diff --git a/include/ADIOS.h b/include/ADIOS.h
index 7765737c32b74be692378f52e6bd42203e036884..f289c278aab493907ae5071133761423c5abb6e3 100644
--- a/include/ADIOS.h
+++ b/include/ADIOS.h
@@ -98,35 +98,20 @@ public:
      * @return
      */
     template <class T>
-    inline Variable<T> &DefineVariable(const std::string &name,
-                                       const Dims dimensions = Dims{1},
-                                       const Dims globalDimensions = Dims(),
-                                       const Dims globalOffsets = Dims())
-    {
-        throw std::invalid_argument("ERROR: type not supported for variable " +
-                                    name + " in call to DefineVariable\n");
-    }
+    Variable<T> &DefineVariable(const std::string &name,
+                                const Dims dimensions = Dims{1},
+                                const Dims globalDimensions = Dims(),
+                                const Dims globalOffsets = Dims());
 
     template <class T>
-    inline Variable<T> &GetVariable(const std::string &name)
-    {
-        throw std::invalid_argument("ERROR: type not supported for variable " +
-                                    name + " in call to GetVariable\n");
-    }
+    Variable<T> &GetVariable(const std::string &name);
 
     template <class T>
-    VariableCompound &DefineVariableCompound(
-        const std::string &name, const Dims dimensions = Dims{1},
-        const Dims globalDimensions = Dims(), const Dims globalOffsets = Dims())
-    {
-        CheckVariableInput(name, dimensions);
-        const unsigned int size = m_Compound.size();
-        m_Compound.emplace(size, VariableCompound(name, sizeof(T), dimensions,
-                                                  globalDimensions,
-                                                  globalOffsets, m_DebugMode));
-        m_Variables.emplace(name, std::make_pair(GetType<T>(), size));
-        return m_Compound.at(size);
-    }
+    VariableCompound &
+    DefineVariableCompound(const std::string &name,
+                           const Dims dimensions = Dims{1},
+                           const Dims globalDimensions = Dims(),
+                           const Dims globalOffsets = Dims());
 
     VariableCompound &GetVariableCompound(const std::string &name);
 
@@ -362,342 +347,28 @@ protected: // no const to allow default empty and copy constructors
                      const std::string &hint) const;
 
     template <class T>
-    unsigned int GetVariableIndex(const std::string &name)
-    {
-        auto itVariable = m_Variables.find(name);
-        CheckVariableName(itVariable, name,
-                          "in call to GetVariable<" + GetType<T>() +
-                              ">, or call to GetVariableCompound if <T> = "
-                              "<compound>\n");
-        return itVariable->second.second;
-    }
-};
-
-// template specializations of DefineVariable:
-template <>
-inline Variable<char> &
-ADIOS::DefineVariable(const std::string &name, const Dims dimensions,
-                      const Dims globalDimensions, const Dims globalOffsets)
-{
-    CheckVariableInput(name, dimensions);
-    const unsigned int size = m_Char.size();
-    m_Char.emplace(size, Variable<char>(name, dimensions, globalDimensions,
-                                        globalOffsets, m_DebugMode));
-    m_Variables.emplace(name, std::make_pair(GetType<char>(), size));
-    return m_Char.at(size);
-}
-
-template <>
-inline Variable<unsigned char> &
-ADIOS::DefineVariable(const std::string &name, const Dims dimensions,
-                      const Dims globalDimensions, const Dims globalOffsets)
-{
-    CheckVariableInput(name, dimensions);
-    const unsigned int size = m_UChar.size();
-    m_UChar.emplace(size,
-                    Variable<unsigned char>(name, dimensions, globalDimensions,
-                                            globalOffsets, m_DebugMode));
-    m_Variables.emplace(name, std::make_pair(GetType<unsigned char>(), size));
-    return m_UChar.at(size);
-}
-
-template <>
-inline Variable<short> &
-ADIOS::DefineVariable(const std::string &name, const Dims dimensions,
-                      const Dims globalDimensions, const Dims globalOffsets)
-{
-    CheckVariableInput(name, dimensions);
-    const unsigned int size = m_Short.size();
-    m_Short.emplace(size, Variable<short>(name, dimensions, globalDimensions,
-                                          globalOffsets, m_DebugMode));
-    m_Variables.emplace(name, std::make_pair(GetType<unsigned char>(), size));
-    return m_Short.at(size);
-}
-
-template <>
-inline Variable<unsigned short> &
-ADIOS::DefineVariable(const std::string &name, const Dims dimensions,
-                      const Dims globalDimensions, const Dims globalOffsets)
-{
-    CheckVariableInput(name, dimensions);
-    const unsigned int size = m_UShort.size();
-    m_UShort.emplace(
-        size, Variable<unsigned short>(name, dimensions, globalDimensions,
-                                       globalOffsets, m_DebugMode));
-    m_Variables.emplace(name, std::make_pair(GetType<unsigned short>(), size));
-    return m_UShort.at(size);
-}
-
-template <>
-inline Variable<int> &
-ADIOS::DefineVariable(const std::string &name, const Dims dimensions,
-                      const Dims globalDimensions, const Dims globalOffsets)
-{
-    CheckVariableInput(name, dimensions);
-    const unsigned int size = m_Int.size();
-    m_Int.emplace(size, Variable<int>(name, dimensions, globalDimensions,
-                                      globalOffsets, m_DebugMode));
-    m_Variables.emplace(name, std::make_pair(GetType<int>(), size));
-    return m_Int.at(size);
-}
-
-template <>
-inline Variable<unsigned int> &
-ADIOS::DefineVariable(const std::string &name, const Dims dimensions,
-                      const Dims globalDimensions, const Dims globalOffsets)
-{
-    CheckVariableInput(name, dimensions);
-    const unsigned int size = m_UInt.size();
-    m_UInt.emplace(size,
-                   Variable<unsigned int>(name, dimensions, globalDimensions,
-                                          globalOffsets, m_DebugMode));
-    m_Variables.emplace(name, std::make_pair(GetType<unsigned int>(), size));
-    return m_UInt.at(size);
-}
-
-template <>
-inline Variable<long int> &
-ADIOS::DefineVariable(const std::string &name, const Dims dimensions,
-                      const Dims globalDimensions, const Dims globalOffsets)
-{
-    CheckVariableInput(name, dimensions);
-    const unsigned int size = m_LInt.size();
-    m_LInt.emplace(size, Variable<long int>(name, dimensions, globalDimensions,
-                                            globalOffsets, m_DebugMode));
-    m_Variables.emplace(name, std::make_pair(GetType<long int>(), size));
-    return m_LInt.at(size);
-}
-
-template <>
-inline Variable<unsigned long int> &
-ADIOS::DefineVariable(const std::string &name, const Dims dimensions,
-                      const Dims globalDimensions, const Dims globalOffsets)
-{
-    CheckVariableInput(name, dimensions);
-    const unsigned int size = m_LInt.size();
-    m_ULInt.emplace(
-        size, Variable<unsigned long int>(name, dimensions, globalDimensions,
-                                          globalOffsets, m_DebugMode));
-    m_Variables.emplace(name,
-                        std::make_pair(GetType<unsigned long int>(), size));
-    return m_ULInt.at(size);
-}
-
-template <>
-inline Variable<long long int> &
-ADIOS::DefineVariable(const std::string &name, const Dims dimensions,
-                      const Dims globalDimensions, const Dims globalOffsets)
-{
-    CheckVariableInput(name, dimensions);
-    const unsigned int size = m_LLInt.size();
-    m_LLInt.emplace(size,
-                    Variable<long long int>(name, dimensions, globalDimensions,
-                                            globalOffsets, m_DebugMode));
-    m_Variables.emplace(name, std::make_pair(GetType<long long int>(), size));
-    return m_LLInt.at(size);
-}
-
-template <>
-inline Variable<unsigned long long int> &
-ADIOS::DefineVariable(const std::string &name, const Dims dimensions,
-                      const Dims globalDimensions, const Dims globalOffsets)
-{
-    CheckVariableInput(name, dimensions);
-    const unsigned int size = m_ULLInt.size();
-    m_ULLInt.emplace(size, Variable<unsigned long long int>(
-                               name, dimensions, globalDimensions,
-                               globalOffsets, m_DebugMode));
-    m_Variables.emplace(
-        name, std::make_pair(GetType<unsigned long long int>(), size));
-    return m_ULLInt.at(size);
-}
-
-template <>
-inline Variable<float> &
-ADIOS::DefineVariable(const std::string &name, const Dims dimensions,
-                      const Dims globalDimensions, const Dims globalOffsets)
-{
-    CheckVariableInput(name, dimensions);
-    const unsigned int size = m_Float.size();
-    m_Float.emplace(size, Variable<float>(name, dimensions, globalDimensions,
-                                          globalOffsets, m_DebugMode));
-    m_Variables.emplace(name, std::make_pair(GetType<float>(), size));
-    return m_Float.at(size);
-}
-
-template <>
-inline Variable<double> &
-ADIOS::DefineVariable(const std::string &name, const Dims dimensions,
-                      const Dims globalDimensions, const Dims globalOffsets)
-{
-    CheckVariableInput(name, dimensions);
-    const unsigned int size = m_Double.size();
-    m_Double.emplace(size, Variable<double>(name, dimensions, globalDimensions,
-                                            globalOffsets, m_DebugMode));
-    m_Variables.emplace(name, std::make_pair(GetType<double>(), size));
-    return m_Double.at(size);
-}
-
-template <>
-inline Variable<long double> &
-ADIOS::DefineVariable(const std::string &name, const Dims dimensions,
-                      const Dims globalDimensions, const Dims globalOffsets)
-{
-    CheckVariableInput(name, dimensions);
-    const unsigned int size = m_LDouble.size();
-    m_LDouble.emplace(size,
-                      Variable<long double>(name, dimensions, globalDimensions,
-                                            globalOffsets, m_DebugMode));
-    m_Variables.emplace(name, std::make_pair(GetType<long double>(), size));
-    return m_LDouble.at(size);
-}
-
-template <>
-inline Variable<std::complex<float>> &
-ADIOS::DefineVariable(const std::string &name, const Dims dimensions,
-                      const Dims globalDimensions, const Dims globalOffsets)
-{
-    CheckVariableInput(name, dimensions);
-    const unsigned int size = m_CFloat.size();
-    m_CFloat.emplace(
-        size, Variable<std::complex<float>>(name, dimensions, globalDimensions,
-                                            globalOffsets, m_DebugMode));
-    m_Variables.emplace(name,
-                        std::make_pair(GetType<std::complex<float>>(), size));
-    return m_CFloat.at(size);
-}
+    unsigned int GetVariableIndex(const std::string &name);
 
-template <>
-inline Variable<std::complex<double>> &
-ADIOS::DefineVariable(const std::string &name, const Dims dimensions,
-                      const Dims globalDimensions, const Dims globalOffsets)
-{
-    CheckVariableInput(name, dimensions);
-    const unsigned int size = m_CDouble.size();
-    m_CDouble.emplace(
-        size, Variable<std::complex<double>>(name, dimensions, globalDimensions,
-                                             globalOffsets, m_DebugMode));
-    m_Variables.emplace(name,
-                        std::make_pair(GetType<std::complex<double>>(), size));
-    return m_CDouble.at(size);
-}
+    // Helper function for DefineVariable
+    template <class T>
+    std::map<unsigned int, Variable<T>> &GetVarMap();
+};
 
-template <>
-inline Variable<std::complex<long double>> &
-ADIOS::DefineVariable(const std::string &name, const Dims dimensions,
-                      const Dims globalDimensions, const Dims globalOffsets)
+template <class T>
+VariableCompound &ADIOS::DefineVariableCompound(const std::string &name,
+                                                const Dims dimensions,
+                                                const Dims globalDimensions,
+                                                const Dims globalOffsets)
 {
     CheckVariableInput(name, dimensions);
-    const unsigned int size = m_CLDouble.size();
-    m_CLDouble.emplace(size, Variable<std::complex<long double>>(
-                                 name, dimensions, globalDimensions,
-                                 globalOffsets, m_DebugMode));
-    m_Variables.emplace(
-        name, std::make_pair(GetType<std::complex<long double>>(), size));
-    return m_CLDouble.at(size);
-}
-
-// Get template specialization
-template <>
-inline Variable<char> &ADIOS::GetVariable(const std::string &name)
-{
-    return m_Char.at(GetVariableIndex<char>(name));
-}
-
-template <>
-inline Variable<unsigned char> &ADIOS::GetVariable(const std::string &name)
-{
-    return m_UChar.at(GetVariableIndex<unsigned char>(name));
-}
-
-template <>
-inline Variable<short> &ADIOS::GetVariable(const std::string &name)
-{
-    return m_Short.at(GetVariableIndex<short>(name));
-}
-
-template <>
-inline Variable<unsigned short> &ADIOS::GetVariable(const std::string &name)
-{
-    return m_UShort.at(GetVariableIndex<unsigned short>(name));
-}
-
-template <>
-inline Variable<int> &ADIOS::GetVariable(const std::string &name)
-{
-    return m_Int.at(GetVariableIndex<int>(name));
-}
-
-template <>
-inline Variable<unsigned int> &ADIOS::GetVariable(const std::string &name)
-{
-    return m_UInt.at(GetVariableIndex<unsigned int>(name));
-}
-
-template <>
-inline Variable<long int> &ADIOS::GetVariable(const std::string &name)
-{
-    return m_LInt.at(GetVariableIndex<unsigned int>(name));
-}
-
-template <>
-inline Variable<unsigned long int> &ADIOS::GetVariable(const std::string &name)
-{
-    return m_ULInt.at(GetVariableIndex<unsigned long int>(name));
-}
-
-template <>
-inline Variable<long long int> &ADIOS::GetVariable(const std::string &name)
-{
-    return m_LLInt.at(GetVariableIndex<long long int>(name));
-}
-
-template <>
-inline Variable<unsigned long long int> &
-ADIOS::GetVariable(const std::string &name)
-{
-    return m_ULLInt.at(GetVariableIndex<unsigned long long int>(name));
-}
-
-template <>
-inline Variable<float> &ADIOS::GetVariable(const std::string &name)
-{
-    return m_Float.at(GetVariableIndex<float>(name));
-}
-
-template <>
-inline Variable<double> &ADIOS::GetVariable(const std::string &name)
-{
-    return m_Double.at(GetVariableIndex<double>(name));
-}
-
-template <>
-inline Variable<long double> &ADIOS::GetVariable(const std::string &name)
-{
-    return m_LDouble.at(GetVariableIndex<long double>(name));
-}
-
-template <>
-inline Variable<std::complex<float>> &
-ADIOS::GetVariable(const std::string &name)
-{
-    return m_CFloat.at(GetVariableIndex<std::complex<float>>(name));
-}
-
-template <>
-inline Variable<std::complex<double>> &
-ADIOS::GetVariable(const std::string &name)
-{
-    return m_CDouble.at(GetVariableIndex<std::complex<double>>(name));
-}
-
-template <>
-inline Variable<std::complex<long double>> &
-ADIOS::GetVariable(const std::string &name)
-{
-    return m_CLDouble.at(GetVariableIndex<std::complex<long double>>(name));
+    const unsigned int size = m_Compound.size();
+    m_Compound.emplace(size, VariableCompound(name, sizeof(T), dimensions,
+                                              globalDimensions, globalOffsets,
+                                              m_DebugMode));
+    m_Variables.emplace(name, std::make_pair(GetType<T>(), size));
+    return m_Compound.at(size);
 }
 
-} // end namespace
+} // end namespace adios
 
 #endif /* ADIOS_H_ */
diff --git a/source/ADIOS.cpp b/source/ADIOS.cpp
index 26718ac35e7e7a4cbbd2d5d875420c87cd7ec469..2d44d18c976555039e0a5abc4082c5585f2c9e3f 100644
--- a/source/ADIOS.cpp
+++ b/source/ADIOS.cpp
@@ -17,6 +17,7 @@
 /// \endcond
 
 #include "ADIOS.h"
+#include "ADIOS.tcc"
 
 #include "functions/adiosFunctions.h"
 
diff --git a/source/ADIOS.tcc b/source/ADIOS.tcc
new file mode 100644
index 0000000000000000000000000000000000000000..ebfc628c6058a4e0761e58b13469f815bfacc0f7
--- /dev/null
+++ b/source/ADIOS.tcc
@@ -0,0 +1,180 @@
+/*
+ * Distributed under the OSI-approved Apache License, Version 2.0.  See
+ * accompanying file Copyright.txt for details.
+ *
+ * ADIOS.tcc
+ *   This contains the template specialization implementations for the ADIOS
+ *   class
+ */
+
+#ifndef ADIOS_TCC_
+#define ADIOS_TCC_
+
+#include <complex>
+#include <map>
+#include <memory> //std::shared_ptr
+#include <ostream>
+#include <set>
+#include <string>
+#include <vector>
+
+#include "ADIOS.h"
+#include "ADIOSMacros.h"
+
+namespace adios
+{
+
+// -----------------------------------------------------------------------------
+// template specializations of GetVarMap helper function
+// -----------------------------------------------------------------------------
+
+template <>
+std::map<unsigned int, Variable<char>> &ADIOS::GetVarMap()
+{
+    return m_Char;
+}
+
+template <>
+std::map<unsigned int, Variable<unsigned char>> &ADIOS::GetVarMap()
+{
+    return m_UChar;
+}
+
+template <>
+std::map<unsigned int, Variable<short>> &ADIOS::GetVarMap()
+{
+    return m_Short;
+}
+
+template <>
+std::map<unsigned int, Variable<unsigned short>> &ADIOS::GetVarMap()
+{
+    return m_UShort;
+}
+
+template <>
+std::map<unsigned int, Variable<int>> &ADIOS::GetVarMap()
+{
+    return m_Int;
+}
+
+template <>
+std::map<unsigned int, Variable<unsigned int>> &ADIOS::GetVarMap()
+{
+    return m_UInt;
+}
+
+template <>
+std::map<unsigned int, Variable<long int>> &ADIOS::GetVarMap()
+{
+    return m_LInt;
+}
+
+template <>
+std::map<unsigned int, Variable<unsigned long int>> &ADIOS::GetVarMap()
+{
+    return m_ULInt;
+}
+
+template <>
+std::map<unsigned int, Variable<long long int>> &ADIOS::GetVarMap()
+{
+    return m_LLInt;
+}
+
+template <>
+std::map<unsigned int, Variable<unsigned long long int>> &ADIOS::GetVarMap()
+{
+    return m_ULLInt;
+}
+
+template <>
+std::map<unsigned int, Variable<float>> &ADIOS::GetVarMap()
+{
+    return m_Float;
+}
+
+template <>
+std::map<unsigned int, Variable<double>> &ADIOS::GetVarMap()
+{
+    return m_Double;
+}
+
+template <>
+std::map<unsigned int, Variable<long double>> &ADIOS::GetVarMap()
+{
+    return m_LDouble;
+}
+
+template <>
+std::map<unsigned int, Variable<std::complex<float>>> &ADIOS::GetVarMap()
+{
+    return m_CFloat;
+}
+
+template <>
+std::map<unsigned int, Variable<std::complex<double>>> &ADIOS::GetVarMap()
+{
+    return m_CDouble;
+}
+
+template <>
+std::map<unsigned int, Variable<std::complex<long double>>> &ADIOS::GetVarMap()
+{
+    return m_CLDouble;
+}
+
+// -----------------------------------------------------------------------------
+// template specializations of DefineVariable:
+// -----------------------------------------------------------------------------
+
+template <typename T>
+Variable<T> &
+ADIOS::DefineVariable(const std::string &name, const Dims dimensions,
+                      const Dims globalDimensions, const Dims globalOffsets)
+{
+    auto &varMap = GetVarMap<T>();
+    CheckVariableInput(name, dimensions);
+    const unsigned int size = varMap.size();
+    varMap.emplace(size, Variable<T>(name, dimensions, globalDimensions,
+                                     globalOffsets, m_DebugMode));
+    m_Variables.emplace(name, std::make_pair(GetType<T>(), size));
+    return varMap.at(size);
+}
+
+#define instantiate_specialization(T)                                          \
+    template Variable<T> &ADIOS::DefineVariable<T>(                            \
+        const std::string &, const Dims, const Dims, const Dims);
+ADIOS_FOREACH_TYPE_1ARG(instantiate_specialization)
+#undef instantiate_specialization
+
+// -----------------------------------------------------------------------------
+// template specializations of DefineVariable:
+// -----------------------------------------------------------------------------
+
+template <class T>
+unsigned int ADIOS::GetVariableIndex(const std::string &name)
+{
+    auto itVariable = m_Variables.find(name);
+    CheckVariableName(
+        itVariable, name,
+        "in call to GetVariable<" + GetType<T>() +
+            ">, or call to GetVariableCompound if <T> = <compound>\n");
+    return itVariable->second.second;
+}
+
+// Get template specialization
+template <typename T>
+Variable<T> &ADIOS::GetVariable(const std::string &name)
+{
+    return GetVarMap<T>().at(GetVariableIndex<T>(name));
+}
+
+#define instantiate_specialization(T)                                          \
+    template Variable<T> &ADIOS::GetVariable<T>(const std::string &);
+ADIOS_FOREACH_TYPE_1ARG(instantiate_specialization)
+#undef instantiate_specialization
+
+} // end namespace adios
+
+#endif /* ADIOS_TCC_ */
diff --git a/source/ADIOSMacros.h b/source/ADIOSMacros.h
new file mode 100644
index 0000000000000000000000000000000000000000..54f8492e9e49f16797d5e52a46e3c0e86d0da600
--- /dev/null
+++ b/source/ADIOSMacros.h
@@ -0,0 +1,42 @@
+/*
+ * Distributed under the OSI-approved Apache License, Version 2.0.  See
+ * accompanying file Copyright.txt for details.
+ *
+ * ADIOSMacros.h
+ *   This contains a set of helper macros used internally
+ */
+#ifndef ADIOSMACROS_H
+#define ADIOSMACROS_H
+
+// The ADIOS_FOREACH_TYPE_1ARG macro assumes the given argument is a macro which
+// takes a single argument that is a type and then inserts the given MACRO for
+// each of the known primitive types
+//
+// An example of this might be to instantiate a template function for each
+// known type.  For example:
+//
+//   template<typename T> int foo() { /* some implementation of foo */ }
+//
+//   #define instantiate_foo(T) template int foo<T>();
+//   ADIOS_FOREACH_TYPE_1ARG(instantiate_foo)
+//   #undef instantiate_foo
+//
+#define ADIOS_FOREACH_TYPE_1ARG(MACRO)                                         \
+    MACRO(char)                                                                \
+    MACRO(unsigned char)                                                       \
+    MACRO(short)                                                               \
+    MACRO(unsigned short)                                                      \
+    MACRO(int)                                                                 \
+    MACRO(unsigned int)                                                        \
+    MACRO(long int)                                                            \
+    MACRO(unsigned long int)                                                   \
+    MACRO(long long int)                                                       \
+    MACRO(unsigned long long int)                                              \
+    MACRO(float)                                                               \
+    MACRO(double)                                                              \
+    MACRO(long double)                                                         \
+    MACRO(std::complex<float>)                                                 \
+    MACRO(std::complex<double>)                                                \
+    MACRO(std::complex<long double>)
+
+#endif // ADIOSMACROS_H