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(); }