diff --git a/Framework/PythonInterface/core/inc/MantidPythonInterface/core/CallMethod.h b/Framework/PythonInterface/core/inc/MantidPythonInterface/core/CallMethod.h
index 5b712b2e972457e8f9da211f477bfaeab8c096bc..928dc14fedd8ca4634c9fc905d4cf39cdf8887c1 100644
--- a/Framework/PythonInterface/core/inc/MantidPythonInterface/core/CallMethod.h
+++ b/Framework/PythonInterface/core/inc/MantidPythonInterface/core/CallMethod.h
@@ -68,6 +68,23 @@ ReturnType callMethodNoCheck(PyObject *obj, const char *methodName,
   return detail::callMethodImpl<ReturnType, Args...>(obj, methodName, args...);
 }
 
+/**
+ * Wrapper around boost::python::call_method to acquire GIL for duration
+ * of call. If the call raises a Python error then this is translated to
+ * a C++ exception object inheriting from std::exception or std::runtime_error
+ * depending on the type of Python error. Overload for boost::python::object
+ * @param obj Reference to boost.python.object wrapper
+ * @param methodName Name of the method call
+ * @param args A list of arguments to forward to call_method
+ */
+template <typename ReturnType, typename... Args>
+ReturnType callMethodNoCheck(const boost::python::object &obj,
+                             const char *methodName, const Args &... args) {
+  GlobalInterpreterLock gil;
+  return detail::callMethodImpl<ReturnType, Args...>(obj.ptr(), methodName,
+                                                     args...);
+}
+
 /**
  * Wrapper around boost::python::call_method to acquire GIL for duration
  * of call. If the attribute does not exist then an UndefinedAttributeError
diff --git a/qt/widgets/mplcpp/inc/MantidQtWidgets/MplCpp/Figure.h b/qt/widgets/mplcpp/inc/MantidQtWidgets/MplCpp/Figure.h
index e1b16e031733ea612a3326446e53bdca11826a89..5e6ab902964edcea2a92552832e488aa1b4cc927 100644
--- a/qt/widgets/mplcpp/inc/MantidQtWidgets/MplCpp/Figure.h
+++ b/qt/widgets/mplcpp/inc/MantidQtWidgets/MplCpp/Figure.h
@@ -7,6 +7,7 @@
 #ifndef MPLCPP_FIGURE_H
 #define MPLCPP_FIGURE_H
 
+#include "MantidPythonInterface/core/GlobalInterpreterLock.h"
 #include "MantidQtWidgets/MplCpp/Axes.h"
 #include "MantidQtWidgets/MplCpp/DllConfig.h"
 #include "MantidQtWidgets/MplCpp/Python/Object.h"
@@ -28,12 +29,16 @@ public:
    * @brief Access (and create if necessar) the active Axes
    * @return An instance of Axes attached to the figure
    */
-  inline Axes gca() const { return Axes{pyobj().attr("gca")()}; }
+  inline Axes gca() const {
+    Mantid::PythonInterface::GlobalInterpreterLock lock;
+    return Axes{pyobj().attr("gca")()};
+  }
   /**
    * @param index The index of the axes to return
    * @return The axes instance
    */
   inline Axes axes(size_t index) const {
+    Mantid::PythonInterface::GlobalInterpreterLock lock;
     return Axes{pyobj().attr("axes")[index]};
   }
 
diff --git a/qt/widgets/mplcpp/inc/MantidQtWidgets/MplCpp/FigureCanvasQt.h b/qt/widgets/mplcpp/inc/MantidQtWidgets/MplCpp/FigureCanvasQt.h
index 215bc3cfc10443980de045a4667fe7dc6d903e4c..5fa06f9415ed2c78202cf697448120b83df5db92 100644
--- a/qt/widgets/mplcpp/inc/MantidQtWidgets/MplCpp/FigureCanvasQt.h
+++ b/qt/widgets/mplcpp/inc/MantidQtWidgets/MplCpp/FigureCanvasQt.h
@@ -40,9 +40,15 @@ public:
   QPointF toDataCoords(QPoint pos) const;
 
   /// Redraw the canvas
-  inline void draw() { pyobj().attr("draw")(); }
+  inline void draw() {
+    Mantid::PythonInterface::GlobalInterpreterLock lock;
+    pyobj().attr("draw")();
+  }
   /// Redraw the canvas if nothing else is happening
-  inline void drawIdle() { pyobj().attr("draw_idle")(); }
+  inline void drawIdle() {
+    Mantid::PythonInterface::GlobalInterpreterLock lock;
+    pyobj().attr("draw_idle")();
+  }
 
 private: // members
   Figure m_figure;
diff --git a/qt/widgets/mplcpp/inc/MantidQtWidgets/MplCpp/Python/Object.h b/qt/widgets/mplcpp/inc/MantidQtWidgets/MplCpp/Python/Object.h
index 91b91f290291e29a6f606d73d2596c99528e09d2..b751ca3a06993952ab8582acea36b157f4a087b2 100644
--- a/qt/widgets/mplcpp/inc/MantidQtWidgets/MplCpp/Python/Object.h
+++ b/qt/widgets/mplcpp/inc/MantidQtWidgets/MplCpp/Python/Object.h
@@ -8,6 +8,7 @@
 #define MPLCPP_PYTHON_OBJECT_H
 
 #include "MantidPythonInterface/core/ErrorHandling.h"
+#include "MantidPythonInterface/core/GlobalInterpreterLock.h"
 #include <boost/python/borrowed.hpp>
 #include <boost/python/dict.hpp>
 #include <boost/python/object.hpp>
@@ -82,6 +83,13 @@ public:
     }
   }
 
+  /// The destructor must hold the GIL to be able reduce the refcount of
+  /// the object
+  ~InstanceHolder() {
+    Mantid::PythonInterface::GlobalInterpreterLock lock;
+    m_instance = Python::Object(); // none
+  }
+
   /// Return the held instance object
   inline const Object &pyobj() const { return m_instance; }
 
diff --git a/qt/widgets/mplcpp/src/Artist.cpp b/qt/widgets/mplcpp/src/Artist.cpp
index a27fa5bd0ce36743ef214f80abf81360cc56eb2d..a003e083228aa3cae341b84ef2c1dcbd29c6ef2b 100644
--- a/qt/widgets/mplcpp/src/Artist.cpp
+++ b/qt/widgets/mplcpp/src/Artist.cpp
@@ -5,8 +5,12 @@
 //     & Institut Laue - Langevin
 // SPDX - License - Identifier: GPL - 3.0 +
 #include "MantidQtWidgets/MplCpp/Artist.h"
+#include "MantidPythonInterface/core/CallMethod.h"
 #include <cassert>
 
+using Mantid::PythonInterface::GlobalInterpreterLock;
+using Mantid::PythonInterface::callMethodNoCheck;
+
 namespace MantidQt {
 namespace Widgets {
 namespace MplCpp {
@@ -22,6 +26,7 @@ Artist::Artist(Python::Object obj) : InstanceHolder(std::move(obj), "draw") {}
  * @param kwargs A dict of known matplotlib.artist.Artist properties
  */
 void Artist::set(Python::Dict kwargs) {
+  GlobalInterpreterLock lock;
   auto args = Python::NewRef(Py_BuildValue("()"));
   pyobj().attr("set")(*args, **kwargs);
 }
@@ -29,7 +34,7 @@ void Artist::set(Python::Dict kwargs) {
 /**
  * Call .remove on the underlying artist
  */
-void Artist::remove() { pyobj().attr("remove")(); }
+void Artist::remove() { callMethodNoCheck<void>(pyobj(), "remove"); }
 
 } // namespace MplCpp
 } // namespace Widgets
diff --git a/qt/widgets/mplcpp/src/Axes.cpp b/qt/widgets/mplcpp/src/Axes.cpp
index 993c730deb0b6c67eee5d5b00f6c042bfa1fe2f0..0107ee4eb42a74da15d0eb663b49613d3e7c53d1 100644
--- a/qt/widgets/mplcpp/src/Axes.cpp
+++ b/qt/widgets/mplcpp/src/Axes.cpp
@@ -5,6 +5,7 @@
 //     & Institut Laue - Langevin
 // SPDX - License - Identifier: GPL - 3.0 +
 #include "MantidQtWidgets/MplCpp/Axes.h"
+#include "MantidPythonInterface/core/CallMethod.h"
 #include "MantidPythonInterface/core/Converters/VectorToNDArray.h"
 #include "MantidPythonInterface/core/Converters/WrapWithNDArray.h"
 #include "MantidPythonInterface/core/ErrorHandling.h"
@@ -16,7 +17,9 @@ namespace MplCpp {
 
 using Mantid::PythonInterface::Converters::VectorToNDArray;
 using Mantid::PythonInterface::Converters::WrapReadOnly;
+using Mantid::PythonInterface::GlobalInterpreterLock;
 using Mantid::PythonInterface::PythonRuntimeError;
+using Mantid::PythonInterface::callMethodNoCheck;
 
 namespace {
 /**
@@ -56,19 +59,25 @@ Axes::Axes(Python::Object obj) : InstanceHolder(std::move(obj), "plot") {}
  * @brief Set the X-axis label
  * @param label String for the axis label
  */
-void Axes::setXLabel(const char *label) { pyobj().attr("set_xlabel")(label); }
+void Axes::setXLabel(const char *label) {
+  callMethodNoCheck<void, const char *>(pyobj(), "set_xlabel", label);
+}
 
 /**
  * @brief Set the Y-axis label
  * @param label String for the axis label
  */
-void Axes::setYLabel(const char *label) { pyobj().attr("set_ylabel")(label); }
+void Axes::setYLabel(const char *label) {
+  callMethodNoCheck<void, const char *>(pyobj(), "set_ylabel", label);
+}
 
 /**
  * @brief Set the title
  * @param label String for the title label
  */
-void Axes::setTitle(const char *label) { pyobj().attr("set_title")(label); }
+void Axes::setTitle(const char *label) {
+  callMethodNoCheck<void, const char *>(pyobj(), "set_title", label);
+}
 
 /**
  * @brief Take the data and draw a single Line2D on the axes
@@ -91,6 +100,7 @@ Line2D Axes::plot(std::vector<double> xdata, std::vector<double> ydata,
   throwIfEmpty(xdata, 'X');
   throwIfEmpty(ydata, 'Y');
 
+  GlobalInterpreterLock lock;
   // Wrap the vector data in a numpy facade to avoid a copy.
   // The vector still owns the data so it needs to be kept alive too
   // It is transferred to the Line2D for this purpose.
@@ -117,6 +127,7 @@ Line2D Axes::plot(std::vector<double> xdata, std::vector<double> ydata,
  */
 Artist Axes::text(double x, double y, QString text,
                   const char *horizontalAlignment) {
+  GlobalInterpreterLock lock;
   auto args =
       Python::NewRef(Py_BuildValue("(ffs)", x, y, text.toLatin1().constData()));
   auto kwargs = Python::NewRef(
@@ -131,16 +142,12 @@ Artist Axes::text(double x, double y, QString text,
  * @raises std::invalid_argument if the value is unknown
  */
 void Axes::setXScale(const char *value) {
-  try {
-    pyobj().attr("set_xscale")(value);
-  } catch (Python::ErrorAlreadySet &) {
-    throw std::invalid_argument(std::string("setXScale: Unknown scale type ") +
-                                value);
-  }
+  callMethodNoCheck<void, const char *>(pyobj(), "set_xscale", value);
 }
 
 /// @return The scale type of the X axis as a string
 QString Axes::getXScale() const {
+  GlobalInterpreterLock lock;
   return toQString(pyobj().attr("get_xscale")());
 }
 
@@ -151,16 +158,12 @@ QString Axes::getXScale() const {
  * @raises std::invalid_argument if the value is unknown
  */
 void Axes::setYScale(const char *value) {
-  try {
-    pyobj().attr("set_yscale")(value);
-  } catch (Python::ErrorAlreadySet &) {
-    throw std::invalid_argument(std::string("setYScale: Unknown scale type ") +
-                                value);
-  }
+  callMethodNoCheck<void, const char *>(pyobj(), "set_yscale", value);
 }
 
 /// @return The scale type of the Y axis as a string
 QString Axes::getYScale() const {
+  GlobalInterpreterLock lock;
   return toQString(pyobj().attr("get_yscale")());
 }
 
@@ -169,6 +172,7 @@ QString Axes::getYScale() const {
  * @return A 2-tuple of (min,max) values for the X axis
  */
 std::tuple<double, double> Axes::getXLim() const {
+  GlobalInterpreterLock lock;
   return limitsToTuple(pyobj(), "get_xlim");
 }
 
@@ -178,7 +182,7 @@ std::tuple<double, double> Axes::getXLim() const {
  * @param max Maximum value
  */
 void Axes::setXLim(double min, double max) const {
-  pyobj().attr("set_xlim")(min, max);
+  callMethodNoCheck<void, double, double>(pyobj(), "set_xlim", min, max);
 }
 
 /**
@@ -186,6 +190,7 @@ void Axes::setXLim(double min, double max) const {
  * @return A 2-tuple of (min,max) values for the Y axis
  */
 std::tuple<double, double> Axes::getYLim() const {
+  GlobalInterpreterLock lock;
   return limitsToTuple(pyobj(), "get_ylim");
 }
 
@@ -195,7 +200,7 @@ std::tuple<double, double> Axes::getYLim() const {
  * @param max Maximum value
  */
 void Axes::setYLim(double min, double max) const {
-  pyobj().attr("set_ylim")(min, max);
+  callMethodNoCheck<void, double, double>(pyobj(), "set_ylim", min, max);
 }
 
 /**
@@ -203,13 +208,17 @@ void Axes::setYLim(double min, double max) const {
  * @param visibleOnly If true then only include visble artists in the
  * calculation
  */
-void Axes::relim(bool visibleOnly) { pyobj().attr("relim")(visibleOnly); }
+void Axes::relim(bool visibleOnly) {
+  callMethodNoCheck<void, bool>(pyobj(), "relim", visibleOnly);
+}
 
 /**
  * Calls Axes.autoscale to enable/disable auto scaling
  * @param enable If true enable autoscaling and perform the automatic rescale
  */
-void Axes::autoscale(bool enable) { pyobj().attr("autoscale")(enable); }
+void Axes::autoscale(bool enable) {
+  callMethodNoCheck<void, bool>(pyobj(), "autoscale", enable);
+}
 
 /**
  * Autoscale the view based on the current data limits. Calls
@@ -219,7 +228,8 @@ void Axes::autoscale(bool enable) { pyobj().attr("autoscale")(enable); }
  * @param scaleY If true (default) scale the Y axis limits
  */
 void Axes::autoscaleView(bool scaleX, bool scaleY) {
-  pyobj().attr("autoscale_view")(Python::Object(), scaleX, scaleY);
+  callMethodNoCheck<void, Python::Object, bool, bool>(
+      pyobj(), "autoscale_view", Python::Object(), scaleX, scaleY);
 }
 
 /**
@@ -231,7 +241,8 @@ void Axes::autoscaleView(bool scaleX, bool scaleY) {
  * @param scaleY If true (default) scale the Y axis limits
  */
 void Axes::autoscaleView(bool tight, bool scaleX, bool scaleY) {
-  pyobj().attr("autoscale_view")(tight, scaleX, scaleY);
+  callMethodNoCheck<void, bool, bool, bool>(pyobj(), "autoscale_view", tight,
+                                            scaleX, scaleY);
 }
 
 } // namespace MplCpp
diff --git a/qt/widgets/mplcpp/src/ColorbarWidget.cpp b/qt/widgets/mplcpp/src/ColorbarWidget.cpp
index 2f39dca08e3d4ac6b40b1564b3e5db6151186678..3b8ea5dc12e5430fc20e6f77cfe1c5a720e17109 100644
--- a/qt/widgets/mplcpp/src/ColorbarWidget.cpp
+++ b/qt/widgets/mplcpp/src/ColorbarWidget.cpp
@@ -5,6 +5,7 @@
 //     & Institut Laue - Langevin
 // SPDX - License - Identifier: GPL - 3.0 +
 #include "MantidQtWidgets/MplCpp/ColorbarWidget.h"
+#include "MantidPythonInterface/core/GlobalInterpreterLock.h"
 #include "MantidQtWidgets/MplCpp/Colors.h"
 #include "MantidQtWidgets/MplCpp/Figure.h"
 #include "MantidQtWidgets/MplCpp/FigureCanvasQt.h"
@@ -14,7 +15,8 @@
 #include <QDoubleValidator>
 #include <QLineEdit>
 #include <QVBoxLayout>
-#include <QVector>
+
+using Mantid::PythonInterface::GlobalInterpreterLock;
 
 namespace MantidQt {
 namespace Widgets {
@@ -256,6 +258,7 @@ void ColorbarWidget::initLayout() {
 void ColorbarWidget::createColorbar(const Python::Object &ticks,
                                     const Python::Object &format) {
   assert(m_canvas);
+  GlobalInterpreterLock lock;
   auto cb = Python::Object(m_mappable.pyobj().attr("colorbar"));
   if (!cb.is_none()) {
     cb.attr("remove")();
diff --git a/qt/widgets/mplcpp/src/Colors.cpp b/qt/widgets/mplcpp/src/Colors.cpp
index 21ede0fa2771807844bc7a1d7efb74f929dbd7f0..28a7dc022379911f779ff89a0eefa426d2551f0a 100644
--- a/qt/widgets/mplcpp/src/Colors.cpp
+++ b/qt/widgets/mplcpp/src/Colors.cpp
@@ -6,7 +6,9 @@
 // SPDX - License - Identifier: GPL - 3.0 +
 #include "MantidQtWidgets/MplCpp/Colors.h"
 #include "MantidPythonInterface/core/ErrorHandling.h"
+#include "MantidPythonInterface/core/GlobalInterpreterLock.h"
 
+using Mantid::PythonInterface::GlobalInterpreterLock;
 using Mantid::PythonInterface::PythonRuntimeError;
 
 namespace MantidQt {
@@ -21,10 +23,33 @@ Python::Object colorsModule() {
   return Python::NewRef(PyImport_ImportModule("matplotlib.colors"));
 }
 
+// Factory function for creating a Normalize instance
+// Holds the GIL
+Python::Object createNormalize(double vmin, double vmax) {
+  GlobalInterpreterLock lock;
+  return colorsModule().attr("Normalize")(vmin, vmax);
+}
+
+// Factory function for creating a SymLogNorm instance
+// Holds the GIL
+Python::Object createSymLog(double linthresh, double linscale, double vmin,
+                            double vmax) {
+  GlobalInterpreterLock lock;
+  return colorsModule().attr("SymLogNorm")(linthresh, linscale, vmin, vmax);
+}
+
+// Factory function for creating a SymLogNorm instance
+// Holds the GIL
+Python::Object createPowerNorm(double gamma, double vmin, double vmax) {
+  GlobalInterpreterLock lock;
+  return colorsModule().attr("PowerNorm")(gamma, vmin, vmax);
+}
+
 /**
  * @return A reference to the matplotlib.ticker module
  */
 Python::Object tickerModule() {
+  GlobalInterpreterLock lock;
   return Python::NewRef(PyImport_ImportModule("matplotlib.ticker"));
 }
 
@@ -32,6 +57,7 @@ Python::Object tickerModule() {
  * @return A reference to the matplotlib.ticker module
  */
 Python::Object scaleModule() {
+  GlobalInterpreterLock lock;
   return Python::NewRef(PyImport_ImportModule("matplotlib.scale"));
 }
 
@@ -54,7 +80,7 @@ NormalizeBase::NormalizeBase(Python::Object obj)
  * @param vmax Maximum value of the data interval
  */
 Normalize::Normalize(double vmin, double vmax)
-    : NormalizeBase(colorsModule().attr("Normalize")(vmin, vmax)) {}
+    : NormalizeBase(createNormalize(vmin, vmax)) {}
 
 // ------------------------ SymLogNorm -----------------------------------------
 /// The threshold below which the scale becomes linear
@@ -75,14 +101,14 @@ double SymLogNorm::DefaultLinearScale = 1.0;
  */
 SymLogNorm::SymLogNorm(double linthresh, double linscale, double vmin,
                        double vmax)
-    : NormalizeBase(
-          colorsModule().attr("SymLogNorm")(linthresh, linscale, vmin, vmax)),
+    : NormalizeBase(createSymLog(linthresh, linscale, vmin, vmax)),
       m_linscale(linscale) {}
 
 /**
  * @return An instance of the SymmetricalLogLocator
  */
 Python::Object SymLogNorm::tickLocator() const {
+  GlobalInterpreterLock lock;
   // Create log transform with base=10
   auto transform = scaleModule().attr("SymmetricalLogTransform")(
       10, Python::Object(pyobj().attr("linthresh")), m_linscale);
@@ -95,6 +121,7 @@ Python::Object SymLogNorm::tickLocator() const {
  * @return
  */
 Python::Object SymLogNorm::labelFormatter() const {
+  GlobalInterpreterLock lock;
   return Python::Object(tickerModule().attr("LogFormatterMathtext")());
 }
 
@@ -108,7 +135,7 @@ Python::Object SymLogNorm::labelFormatter() const {
  * @param vmax Maximum value of the data interval
  */
 PowerNorm::PowerNorm(double gamma, double vmin, double vmax)
-    : NormalizeBase(colorsModule().attr("PowerNorm")(gamma, vmin, vmax)) {}
+    : NormalizeBase(createPowerNorm(gamma, vmin, vmax)) {}
 
 } // namespace MplCpp
 } // namespace Widgets
diff --git a/qt/widgets/mplcpp/src/Cycler.cpp b/qt/widgets/mplcpp/src/Cycler.cpp
index dc5f1a265ca1eee5dba0b757f46c88a89b937d4c..0bb72af4c98a89102e037a59053a750bf3c49ecf 100644
--- a/qt/widgets/mplcpp/src/Cycler.cpp
+++ b/qt/widgets/mplcpp/src/Cycler.cpp
@@ -5,6 +5,9 @@
 //     & Institut Laue - Langevin
 // SPDX - License - Identifier: GPL - 3.0 +
 #include "MantidQtWidgets/MplCpp/Cycler.h"
+#include "MantidPythonInterface/core/GlobalInterpreterLock.h"
+
+using Mantid::PythonInterface::GlobalInterpreterLock;
 
 namespace MantidQt {
 namespace Widgets {
@@ -21,10 +24,10 @@ Python::Object cyclerModule() {
  * @return An iterable returned from itertools.cycle
  */
 Python::Object cycleIterator(const Python::Object &rawCycler) {
+  GlobalInterpreterLock lock;
   auto itertools = Python::NewRef(PyImport_ImportModule("itertools"));
   try {
     return Python::Object(itertools.attr("cycle")(rawCycler));
-
   } catch (Python::ErrorAlreadySet &) {
     throw std::invalid_argument("itertools.cycle() - Object not iterable");
   }
@@ -44,6 +47,7 @@ Cycler::Cycler(Python::Object obj)
  * @return The next item in the cycle
  */
 Python::Dict Cycler::operator()() const {
+  GlobalInterpreterLock lock;
   return Python::Dict(pyobj().attr("next")());
 }
 
@@ -54,6 +58,7 @@ Python::Dict Cycler::operator()() const {
  * @return A new cycler object wrapping the given iterable sequence
  */
 Cycler cycler(const char *label, const char *iterable) {
+  GlobalInterpreterLock lock;
   return cyclerModule().attr("cycler")(label, iterable);
 }
 
diff --git a/qt/widgets/mplcpp/src/Figure.cpp b/qt/widgets/mplcpp/src/Figure.cpp
index 5977696ba109b91e1c6ed291cb12b950666a7376..f9bdbf90e5517d68b144cfc10a18012f841f3a5d 100644
--- a/qt/widgets/mplcpp/src/Figure.cpp
+++ b/qt/widgets/mplcpp/src/Figure.cpp
@@ -5,6 +5,10 @@
 //     & Institut Laue - Langevin
 // SPDX - License - Identifier: GPL - 3.0 +
 #include "MantidQtWidgets/MplCpp/Figure.h"
+#include "MantidPythonInterface/core/CallMethod.h"
+
+using Mantid::PythonInterface::GlobalInterpreterLock;
+using Mantid::PythonInterface::callMethodNoCheck;
 
 namespace MantidQt {
 namespace Widgets {
@@ -12,6 +16,7 @@ namespace MplCpp {
 
 namespace {
 Python::Object newFigure(bool tightLayout = true) {
+  GlobalInterpreterLock lock;
   Python::Object figureModule{
       Python::NewRef(PyImport_ImportModule("matplotlib.figure"))};
   auto fig = figureModule.attr("Figure")();
@@ -27,7 +32,8 @@ Python::Object newFigure(bool tightLayout = true) {
  * Construct a C++ wrapper around an existing figure instance
  * @param obj An existing Figure instance
  */
-Figure::Figure(Python::Object obj) : Python::InstanceHolder(obj, "add_axes") {}
+Figure::Figure(Python::Object obj)
+    : Python::InstanceHolder(std::move(obj), "add_axes") {}
 
 /**
  * Construct a new default figure.
@@ -42,7 +48,7 @@ Figure::Figure(bool tightLayout)
  * See https://matplotlib.org/api/colors_api.html
  */
 void Figure::setFaceColor(const char *color) {
-  pyobj().attr("set_facecolor")(color);
+  callMethodNoCheck<void, const char *>(pyobj(), "set_facecolor", color);
 }
 
 /**
@@ -55,6 +61,7 @@ void Figure::setFaceColor(const char *color) {
  * @return A new Axes instance
  */
 Axes Figure::addAxes(double left, double bottom, double width, double height) {
+  GlobalInterpreterLock lock;
   return Axes{pyobj().attr("add_axes")(
       Python::NewRef(Py_BuildValue("(ffff)", left, bottom, width, height)))};
 }
@@ -65,6 +72,7 @@ Axes Figure::addAxes(double left, double bottom, double width, double height) {
  * @return
  */
 Axes Figure::addSubPlot(int subplotspec) {
+  GlobalInterpreterLock lock;
   return Axes{pyobj().attr("add_subplot")(subplotspec)};
 }
 
@@ -80,6 +88,7 @@ Axes Figure::addSubPlot(int subplotspec) {
 Python::Object Figure::colorbar(const ScalarMappable &mappable, const Axes &cax,
                                 const Python::Object &ticks,
                                 const Python::Object &format) {
+  GlobalInterpreterLock lock;
   auto args = Python::NewRef(
       Py_BuildValue("(OO)", mappable.pyobj().ptr(), cax.pyobj().ptr()));
   auto kwargs = Python::NewRef(
diff --git a/qt/widgets/mplcpp/src/FigureCanvasQt.cpp b/qt/widgets/mplcpp/src/FigureCanvasQt.cpp
index 28198fed32b842747c61caaebdcc94199b9ae977..19f3604a2160287ba911a43a77f243d6eff6684f 100644
--- a/qt/widgets/mplcpp/src/FigureCanvasQt.cpp
+++ b/qt/widgets/mplcpp/src/FigureCanvasQt.cpp
@@ -10,10 +10,12 @@
 #include "MantidQtWidgets/MplCpp/Python/Sip.h"
 
 #include "MantidPythonInterface/core/ErrorHandling.h"
+#include "MantidPythonInterface/core/GlobalInterpreterLock.h"
 #include "MantidPythonInterface/core/NDArray.h"
 
 #include <QVBoxLayout>
 
+using Mantid::PythonInterface::GlobalInterpreterLock;
 using Mantid::PythonInterface::NDArray;
 
 namespace MantidQt {
@@ -28,6 +30,7 @@ const char *DEFAULT_FACECOLOR = "w";
  * @return A new FigureCanvasQT object
  */
 Python::Object createPyCanvasFromFigure(Figure fig) {
+  GlobalInterpreterLock lock;
   return backendModule().attr("FigureCanvasQTAgg")(fig.pyobj());
 }
 
@@ -86,6 +89,7 @@ QPointF FigureCanvasQt::toDataCoords(QPoint pos) const {
   // and then to the data coordinates
   const int dpiRatio(devicePixelRatio());
   const double xPosPhysical = pos.x() * dpiRatio;
+  GlobalInterpreterLock lock;
   // Y=0 is at the bottom
   double height = PyFloat_AsDouble(
       Python::Object(m_figure.pyobj().attr("bbox").attr("height")).ptr());
diff --git a/qt/widgets/mplcpp/src/ScalarMappable.cpp b/qt/widgets/mplcpp/src/ScalarMappable.cpp
index 2d445a640b4b068eee41afe6d368e995d5d415ba..74d3c31e5e6b1746047877b0589190ba4c051c83 100644
--- a/qt/widgets/mplcpp/src/ScalarMappable.cpp
+++ b/qt/widgets/mplcpp/src/ScalarMappable.cpp
@@ -7,15 +7,16 @@
 #include "MantidQtWidgets/MplCpp/ScalarMappable.h"
 #include "MantidQtWidgets/MplCpp/Colormap.h"
 
+#include "MantidPythonInterface/core/CallMethod.h"
 #include "MantidPythonInterface/core/Converters/VectorToNDArray.h"
 #include "MantidPythonInterface/core/Converters/WrapWithNDArray.h"
 #include "MantidPythonInterface/core/NDArray.h"
 
-#include <iostream>
-
 using Mantid::PythonInterface::Converters::VectorToNDArray;
 using Mantid::PythonInterface::Converters::WrapReadOnly;
+using Mantid::PythonInterface::GlobalInterpreterLock;
 using Mantid::PythonInterface::NDArray;
+using Mantid::PythonInterface::callMethodNoCheck;
 
 namespace MantidQt {
 namespace Widgets {
@@ -57,7 +58,7 @@ ScalarMappable::ScalarMappable(const NormalizeBase &norm, const QString &cmap)
  * @param cmap An instance of a Colormap
  */
 void ScalarMappable::setCmap(const Colormap &cmap) {
-  pyobj().attr("set_cmap")(cmap.pyobj());
+  callMethodNoCheck<void, Python::Object>(pyobj(), "set_cmap", cmap.pyobj());
 }
 
 /**
@@ -65,7 +66,8 @@ void ScalarMappable::setCmap(const Colormap &cmap) {
  * @param cmap The name of a colormap
  */
 void ScalarMappable::setCmap(const QString &cmap) {
-  pyobj().attr("set_cmap")(cmap.toLatin1().constData());
+  callMethodNoCheck<void, const char *>(pyobj(), "set_cmap",
+                                        cmap.toLatin1().constData());
 }
 
 /**
@@ -73,7 +75,7 @@ void ScalarMappable::setCmap(const QString &cmap) {
  * @param norm A normalization type
  */
 void ScalarMappable::setNorm(const NormalizeBase &norm) {
-  pyobj().attr("set_norm")(norm.pyobj());
+  callMethodNoCheck<void, Python::Object>(pyobj(), "set_norm", norm.pyobj());
 }
 
 /**
@@ -83,6 +85,7 @@ void ScalarMappable::setNorm(const NormalizeBase &norm) {
  */
 void ScalarMappable::setClim(boost::optional<double> vmin,
                              boost::optional<double> vmax) {
+  GlobalInterpreterLock lock;
   Python::Object none;
   auto setClimAttr = pyobj().attr("set_clim");
   if (vmin.is_initialized() && vmax.is_initialized()) {
@@ -101,6 +104,7 @@ void ScalarMappable::setClim(boost::optional<double> vmin,
  * @return A QRgb value corresponding to this data point
  */
 QRgb ScalarMappable::toRGBA(double x, double alpha) const {
+  GlobalInterpreterLock lock;
   return toRGBA(std::vector<double>(1, x), alpha)[0];
 }
 
@@ -113,6 +117,7 @@ QRgb ScalarMappable::toRGBA(double x, double alpha) const {
 std::vector<QRgb> ScalarMappable::toRGBA(const std::vector<double> &x,
                                          double alpha) const {
   std::vector<QRgb> rgbaVector(x.size());
+  GlobalInterpreterLock lock;
   auto ndarrayView = Python::NewRef(VectorToNDArray<double, WrapReadOnly>()(x));
   // The final argument (bytes=true) forces the return value to be 0->255
   NDArray bytes{pyobj().attr("to_rgba")(ndarrayView, alpha, true)};
diff --git a/qt/widgets/mplcpp/test/MplCppTestInitialization.h b/qt/widgets/mplcpp/test/MplCppTestInitialization.h
index 6a27c4f0937f520067876d10d6bdd9302648de43..b881938f139f4802fafaa2968330047723f10187 100644
--- a/qt/widgets/mplcpp/test/MplCppTestInitialization.h
+++ b/qt/widgets/mplcpp/test/MplCppTestInitialization.h
@@ -23,6 +23,7 @@ class PythonInterpreter : CxxTest::GlobalFixture {
 public:
   bool setUpWorld() override {
     Py_Initialize();
+    PyEval_InitThreads();
     Mantid::PythonInterface::importNumpy();
     return Py_IsInitialized();
   }