Skip to content

Commit

Permalink
Document the new module_ constructor and add an extensive note to the…
Browse files Browse the repository at this point in the history
… upgrade guide
  • Loading branch information
YannickJadoul committed Feb 1, 2021
1 parent 7f64ed3 commit 126da01
Show file tree
Hide file tree
Showing 2 changed files with 50 additions and 1 deletion.
39 changes: 39 additions & 0 deletions docs/upgrade.rst
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,45 @@ to a new version. But it goes into more detail. This includes things like
deprecated APIs and their replacements, build system changes, general code
modernization and other useful information.

v2.7
====

The previously-deprecated constructor of ``py::module_`` can now be used to
create a module that is *not* used as the top-level module of a C extension.
Previous use of the deprecated ``PYBIND11_PLUGIN`` macro without using
``py::module_::create_extension_module`` will result in the following error
on import: ``SystemError: initialization of example did not return an extension module``.

In this case, change the following double-deprecated, memory-leaking pattern:

.. code-block:: cpp
PYBIND11_PLUGIN(example) {
auto m = pybind11::module_("example", "pybind11 example plugin");
/// Set up bindings here
return m.ptr();
}
into this:

.. code-block:: cpp
static pybind11::module_::module_def example_module_def;
PYBIND11_PLUGIN(example) {
auto m = pybind11::module_::create_extension_module(
"example", "pybind11 example plugin", &example_module_def);
/// Set up bindings here
return m.ptr();
}
or, preferably, avoid the deprecated ``PYBIND11_PLUGIN`` completely:

.. code-block:: cpp
PYBIND11_MODULE(example, m) {
/// Set up bindings here
}
.. _upgrade-guide-2.6:

v2.6
Expand Down
12 changes: 11 additions & 1 deletion include/pybind11/pybind11.h
Original file line number Diff line number Diff line change
Expand Up @@ -892,7 +892,17 @@ class module_ : public object {
public:
PYBIND11_OBJECT_DEFAULT(module_, object, PyModule_Check)

/// Create a new top-level Python module with the given name and docstring
/** \rst
Create a new top-level Python module with the given name and docstring.
Note that this cannot be used as the top-level module for a C extension.
Use `module_::create_extension_module` to create a module object that is
accepted by Python as top-level C extension module, returned from the
extension's ``PyInit_foo`` entry point (``initfoo`` in Python 2).
If you do *not* need a top-level C extension module, this constructor has
the advantage of not needing a ``PyModuleDef`` and thus being easier to use
w.r.t. memory management.
\endrst */
explicit module_(const char *name, const char *doc = nullptr) {
#if defined(PYPY_VERSION) && (PYPY_VERSION_NUM < 0x07030400)
m_ptr = PyModule_New(const_cast<char *>(name));
Expand Down

0 comments on commit 126da01

Please sign in to comment.