Skip to content

Commit a3b300e

Browse files
committed
Merge branch 'master' into sh_merge_master
2 parents a05a201 + ef5a956 commit a3b300e

21 files changed

+537
-19
lines changed

.github/workflows/ci.yml

+1-1
Original file line numberDiff line numberDiff line change
@@ -271,7 +271,7 @@ jobs:
271271
- uses: actions/checkout@v4
272272

273273
- name: Setup Python ${{ matrix.python-version }} (deadsnakes)
274-
uses: deadsnakes/action@v3.1.0
274+
uses: deadsnakes/action@v3.2.0
275275
with:
276276
python-version: ${{ matrix.python-version }}
277277
debug: ${{ matrix.python-debug }}

.github/workflows/pip.yml

+1-1
Original file line numberDiff line numberDiff line change
@@ -103,7 +103,7 @@ jobs:
103103
- uses: actions/download-artifact@v4
104104

105105
- name: Generate artifact attestation for sdist and wheel
106-
uses: actions/attest-build-provenance@6149ea5740be74af77f260b9db67e633f6b0a9a1 # v1.4.2
106+
uses: actions/attest-build-provenance@1c608d11d69870c2092266b3f9a6f3abbf17002c # v1.4.3
107107
with:
108108
subject-path: "*/pybind11*"
109109

.pre-commit-config.yaml

+5-5
Original file line numberDiff line numberDiff line change
@@ -32,15 +32,15 @@ repos:
3232

3333
# Ruff, the Python auto-correcting linter/formatter written in Rust
3434
- repo: https://github.com/astral-sh/ruff-pre-commit
35-
rev: v0.5.6
35+
rev: v0.6.3
3636
hooks:
3737
- id: ruff
3838
args: ["--fix", "--show-fixes"]
3939
- id: ruff-format
4040

4141
# Check static types with mypy
4242
- repo: https://github.com/pre-commit/mirrors-mypy
43-
rev: "v1.11.1"
43+
rev: "v1.11.2"
4444
hooks:
4545
- id: mypy
4646
args: []
@@ -95,7 +95,7 @@ repos:
9595

9696
# Avoid directional quotes
9797
- repo: https://github.com/sirosen/texthooks
98-
rev: "0.6.6"
98+
rev: "0.6.7"
9999
hooks:
100100
- id: fix-ligatures
101101
- id: fix-smartquotes
@@ -144,14 +144,14 @@ repos:
144144

145145
# PyLint has native support - not always usable, but works for us
146146
- repo: https://github.com/PyCQA/pylint
147-
rev: "v3.2.6"
147+
rev: "v3.2.7"
148148
hooks:
149149
- id: pylint
150150
files: ^pybind11
151151

152152
# Check schemas on some of our YAML files
153153
- repo: https://github.com/python-jsonschema/check-jsonschema
154-
rev: 0.29.1
154+
rev: 0.29.2
155155
hooks:
156156
- id: check-readthedocs
157157
- id: check-github-workflows

CMakeLists.txt

+1
Original file line numberDiff line numberDiff line change
@@ -129,6 +129,7 @@ endif()
129129
set(PYBIND11_HEADERS
130130
include/pybind11/detail/class.h
131131
include/pybind11/detail/common.h
132+
include/pybind11/detail/cpp_conduit.h
132133
include/pybind11/detail/descr.h
133134
include/pybind11/detail/dynamic_raw_ptr_cast_if_possible.h
134135
include/pybind11/detail/init.h

include/pybind11/detail/cpp_conduit.h

+77
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,77 @@
1+
// Copyright (c) 2024 The pybind Community.
2+
3+
#pragma once
4+
5+
#include <pybind11/pytypes.h>
6+
7+
#include "common.h"
8+
#include "internals.h"
9+
10+
#include <typeinfo>
11+
12+
PYBIND11_NAMESPACE_BEGIN(PYBIND11_NAMESPACE)
13+
PYBIND11_NAMESPACE_BEGIN(detail)
14+
15+
// Forward declaration needed here: Refactoring opportunity.
16+
extern "C" inline PyObject *pybind11_object_new(PyTypeObject *type, PyObject *, PyObject *);
17+
18+
inline bool type_is_managed_by_our_internals(PyTypeObject *type_obj) {
19+
#if defined(PYPY_VERSION)
20+
auto &internals = get_internals();
21+
return bool(internals.registered_types_py.find(type_obj)
22+
!= internals.registered_types_py.end());
23+
#else
24+
return bool(type_obj->tp_new == pybind11_object_new);
25+
#endif
26+
}
27+
28+
inline bool is_instance_method_of_type(PyTypeObject *type_obj, PyObject *attr_name) {
29+
PyObject *descr = _PyType_Lookup(type_obj, attr_name);
30+
return bool((descr != nullptr) && PyInstanceMethod_Check(descr));
31+
}
32+
33+
inline object try_get_cpp_conduit_method(PyObject *obj) {
34+
if (PyType_Check(obj)) {
35+
return object();
36+
}
37+
PyTypeObject *type_obj = Py_TYPE(obj);
38+
str attr_name("_pybind11_conduit_v1_");
39+
bool assumed_to_be_callable = false;
40+
if (type_is_managed_by_our_internals(type_obj)) {
41+
if (!is_instance_method_of_type(type_obj, attr_name.ptr())) {
42+
return object();
43+
}
44+
assumed_to_be_callable = true;
45+
}
46+
PyObject *method = PyObject_GetAttr(obj, attr_name.ptr());
47+
if (method == nullptr) {
48+
PyErr_Clear();
49+
return object();
50+
}
51+
if (!assumed_to_be_callable && PyCallable_Check(method) == 0) {
52+
Py_DECREF(method);
53+
return object();
54+
}
55+
return reinterpret_steal<object>(method);
56+
}
57+
58+
inline void *try_raw_pointer_ephemeral_from_cpp_conduit(handle src,
59+
const std::type_info *cpp_type_info) {
60+
object method = try_get_cpp_conduit_method(src.ptr());
61+
if (method) {
62+
capsule cpp_type_info_capsule(const_cast<void *>(static_cast<const void *>(cpp_type_info)),
63+
typeid(std::type_info).name());
64+
object cpp_conduit = method(bytes(PYBIND11_PLATFORM_ABI_ID),
65+
cpp_type_info_capsule,
66+
bytes("raw_pointer_ephemeral"));
67+
if (isinstance<capsule>(cpp_conduit)) {
68+
return reinterpret_borrow<capsule>(cpp_conduit).get_pointer();
69+
}
70+
}
71+
return nullptr;
72+
}
73+
74+
#define PYBIND11_HAS_CPP_CONDUIT 1
75+
76+
PYBIND11_NAMESPACE_END(detail)
77+
PYBIND11_NAMESPACE_END(PYBIND11_NAMESPACE)

include/pybind11/detail/internals.h

+6-4
Original file line numberDiff line numberDiff line change
@@ -347,15 +347,17 @@ struct type_info {
347347
# define PYBIND11_INTERNALS_KIND ""
348348
#endif
349349

350+
#define PYBIND11_PLATFORM_ABI_ID \
351+
PYBIND11_INTERNALS_KIND PYBIND11_COMPILER_TYPE PYBIND11_STDLIB PYBIND11_BUILD_ABI \
352+
PYBIND11_BUILD_TYPE
353+
350354
#define PYBIND11_INTERNALS_ID \
351355
"__pybind11_internals_v" PYBIND11_TOSTRING(PYBIND11_INTERNALS_VERSION) \
352-
PYBIND11_INTERNALS_KIND PYBIND11_COMPILER_TYPE PYBIND11_STDLIB \
353-
PYBIND11_BUILD_ABI PYBIND11_BUILD_TYPE "__"
356+
PYBIND11_PLATFORM_ABI_ID "__"
354357

355358
#define PYBIND11_MODULE_LOCAL_ID \
356359
"__pybind11_module_local_v" PYBIND11_TOSTRING(PYBIND11_INTERNALS_VERSION) \
357-
PYBIND11_INTERNALS_KIND PYBIND11_COMPILER_TYPE PYBIND11_STDLIB \
358-
PYBIND11_BUILD_ABI PYBIND11_BUILD_TYPE "__"
360+
PYBIND11_PLATFORM_ABI_ID "__"
359361

360362
/// Each module locally stores a pointer to the `internals` data. The data
361363
/// itself is shared among modules with the same `PYBIND11_INTERNALS_ID`.

include/pybind11/detail/type_caster_base.h

+40
Original file line numberDiff line numberDiff line change
@@ -14,6 +14,7 @@
1414
#include <pybind11/trampoline_self_life_support.h>
1515

1616
#include "common.h"
17+
#include "cpp_conduit.h"
1718
#include "descr.h"
1819
#include "dynamic_raw_ptr_cast_if_possible.h"
1920
#include "internals.h"
@@ -22,8 +23,10 @@
2223
#include "value_and_holder.h"
2324

2425
#include <cstdint>
26+
#include <cstring>
2527
#include <iterator>
2628
#include <new>
29+
#include <stdexcept>
2730
#include <string>
2831
#include <type_traits>
2932
#include <typeindex>
@@ -985,6 +988,13 @@ class type_caster_generic {
985988
}
986989
return false;
987990
}
991+
bool try_cpp_conduit(handle src) {
992+
value = try_raw_pointer_ephemeral_from_cpp_conduit(src, cpptype);
993+
if (value != nullptr) {
994+
return true;
995+
}
996+
return false;
997+
}
988998
void check_holder_compat() {}
989999

9901000
PYBIND11_NOINLINE static void *local_load(PyObject *src, const type_info *ti) {
@@ -1116,6 +1126,10 @@ class type_caster_generic {
11161126
return true;
11171127
}
11181128

1129+
if (convert && cpptype && this_.try_cpp_conduit(src)) {
1130+
return true;
1131+
}
1132+
11191133
return false;
11201134
}
11211135

@@ -1143,6 +1157,32 @@ class type_caster_generic {
11431157
void *value = nullptr;
11441158
};
11451159

1160+
inline object cpp_conduit_method(handle self,
1161+
const bytes &pybind11_platform_abi_id,
1162+
const capsule &cpp_type_info_capsule,
1163+
const bytes &pointer_kind) {
1164+
#ifdef PYBIND11_HAS_STRING_VIEW
1165+
using cpp_str = std::string_view;
1166+
#else
1167+
using cpp_str = std::string;
1168+
#endif
1169+
if (cpp_str(pybind11_platform_abi_id) != PYBIND11_PLATFORM_ABI_ID) {
1170+
return none();
1171+
}
1172+
if (std::strcmp(cpp_type_info_capsule.name(), typeid(std::type_info).name()) != 0) {
1173+
return none();
1174+
}
1175+
if (cpp_str(pointer_kind) != "raw_pointer_ephemeral") {
1176+
throw std::runtime_error("Invalid pointer_kind: \"" + std::string(pointer_kind) + "\"");
1177+
}
1178+
const auto *cpp_type_info = cpp_type_info_capsule.get_pointer<const std::type_info>();
1179+
type_caster_generic caster(*cpp_type_info);
1180+
if (!caster.load(self, false)) {
1181+
return none();
1182+
}
1183+
return capsule(caster.value, cpp_type_info->name());
1184+
}
1185+
11461186
/**
11471187
* Determine suitable casting operator for pointer-or-lvalue-casting type casters. The type caster
11481188
* needs to provide `operator T*()` and `operator T&()` operators.

include/pybind11/pybind11.h

+5-2
Original file line numberDiff line numberDiff line change
@@ -594,7 +594,8 @@ class cpp_function : public function {
594594
int index = 0;
595595
/* Create a nice pydoc rec including all signatures and
596596
docstrings of the functions in the overload chain */
597-
if (chain && options::show_function_signatures()) {
597+
if (chain && options::show_function_signatures()
598+
&& std::strcmp(rec->name, "_pybind11_conduit_v1_") != 0) {
598599
// First a generic signature
599600
signatures += rec->name;
600601
signatures += "(*args, **kwargs)\n";
@@ -603,7 +604,8 @@ class cpp_function : public function {
603604
// Then specific overload signatures
604605
bool first_user_def = true;
605606
for (auto *it = chain_start; it != nullptr; it = it->next) {
606-
if (options::show_function_signatures()) {
607+
if (options::show_function_signatures()
608+
&& std::strcmp(rec->name, "_pybind11_conduit_v1_") != 0) {
607609
if (index > 0) {
608610
signatures += '\n';
609611
}
@@ -1859,6 +1861,7 @@ class class_ : public detail::generic_type {
18591861
= instances[std::type_index(typeid(type))];
18601862
});
18611863
}
1864+
def("_pybind11_conduit_v1_", cpp_conduit_method);
18621865
}
18631866

18641867
template <typename Base, detail::enable_if_t<is_base<Base>::value, int> = 0>

tests/CMakeLists.txt

+3
Original file line numberDiff line numberDiff line change
@@ -135,6 +135,7 @@ set(PYBIND11_TEST_FILES
135135
test_const_name
136136
test_constants_and_functions
137137
test_copy_move
138+
test_cpp_conduit
138139
test_custom_type_casters
139140
test_custom_type_setup
140141
test_docstring_options
@@ -235,6 +236,8 @@ tests_extra_targets("test_exceptions.py;test_local_bindings.py;test_stl.py;test_
235236
# And add additional targets for other tests.
236237
tests_extra_targets("test_exceptions.py" "cross_module_interleaved_error_already_set")
237238
tests_extra_targets("test_gil_scoped.py" "cross_module_gil_utils")
239+
tests_extra_targets("test_cpp_conduit.py"
240+
"exo_planet_pybind11;exo_planet_c_api;home_planet_very_lonely_traveler")
238241

239242
set(PYBIND11_EIGEN_REPO
240243
"https://gitlab.com/libeigen/eigen.git"

tests/conftest.py

+3-3
Original file line numberDiff line numberDiff line change
@@ -136,7 +136,7 @@ def stderr(self):
136136
return Output(self.err)
137137

138138

139-
@pytest.fixture()
139+
@pytest.fixture
140140
def capture(capsys):
141141
"""Extended `capsys` with context manager and custom equality operators"""
142142
return Capture(capsys)
@@ -172,7 +172,7 @@ def _sanitize_docstring(thing):
172172
return _sanitize_general(s)
173173

174174

175-
@pytest.fixture()
175+
@pytest.fixture
176176
def doc():
177177
"""Sanitize docstrings and add custom failure explanation"""
178178
return SanitizedString(_sanitize_docstring)
@@ -184,7 +184,7 @@ def _sanitize_message(thing):
184184
return _hexadecimal.sub("0", s)
185185

186186

187-
@pytest.fixture()
187+
@pytest.fixture
188188
def msg():
189189
"""Sanitize messages and add custom failure explanation"""
190190
return SanitizedString(_sanitize_message)

0 commit comments

Comments
 (0)