Skip to content

Commit

Permalink
changes from review
Browse files Browse the repository at this point in the history
  • Loading branch information
rath3t committed Aug 20, 2024
1 parent 0235533 commit 9768d5f
Show file tree
Hide file tree
Showing 2 changed files with 29 additions and 16 deletions.
35 changes: 19 additions & 16 deletions include/pybind11/functional.h
Original file line number Diff line number Diff line change
Expand Up @@ -102,8 +102,8 @@ struct type_caster<std::function<Return(Args...)>> {
rec = c.get_pointer<function_record>();
}
while (rec != nullptr) {
const int correctingSelfArgument = rec->is_method ? 1 : 0;
if (rec->nargs - correctingSelfArgument != sizeof...(Args)) {
const size_t self_offset = rec->is_method ? 1 : 0;
if (rec->nargs != sizeof...(Args) + self_offset) {
rec = rec->next;
// if the overload is not feasible in terms of number of arguments, we
// continue to the next one. If there is no next one, we return false.
Expand All @@ -129,31 +129,34 @@ struct type_caster<std::function<Return(Args...)>> {
// See PR #1413 for full details
} else {
// Check number of arguments of Python function
auto getArgCount = [&](PyObject *obj) {
// This is faster then doing import inspect and inspect.signature(obj).parameters
auto *t = PyObject_GetAttrString(obj, "__code__");
auto *argCount = PyObject_GetAttrString(t, "co_argcount");
return PyLong_AsLong(argCount);
auto argCountFromFuncCode = [&](handle &obj) {
// This is faster then doing import inspect and
// inspect.signature(obj).parameters

object argCount = obj.attr("co_argcount");
return argCount.template cast<size_t>();
};
long argCount = -1;
size_t argCount = 0;

if (static_cast<bool>(PyObject_HasAttrString(src.ptr(), "__code__"))) {
argCount = getArgCount(src.ptr());
handle codeAttr = PyObject_GetAttrString(src.ptr(), "__code__");
if (codeAttr) {
argCount = argCountFromFuncCode(codeAttr);
} else {
if (static_cast<bool>(PyObject_HasAttrString(src.ptr(), "__call__"))) {
auto *t2 = PyObject_GetAttrString(src.ptr(), "__call__");
argCount = getArgCount(t2) - 1; // we have to remove the self argument
handle callAttr = PyObject_GetAttrString(src.ptr(), "__call__");
if (callAttr) {
handle codeAttr2 = callAttr.attr("__code__");
argCount = argCountFromFuncCode(codeAttr2)
- 1; // we have to remove the self argument
} else {
// No __code__ or __call__ attribute, this is not a proper Python function
return false;
}
}
// if we are a method, we have to correct the argument count since we are not counting
// the self argument
const int correctingSelfArgument
= static_cast<bool>(PyMethod_Check(src.ptr())) ? 1 : 0;
const size_t self_offset = static_cast<bool>(PyMethod_Check(src.ptr())) ? 1 : 0;

argCount -= correctingSelfArgument;
argCount -= self_offset;
if (argCount != sizeof...(Args)) {
return false;
}
Expand Down
10 changes: 10 additions & 0 deletions tests/test_callbacks.py
Original file line number Diff line number Diff line change
Expand Up @@ -107,13 +107,23 @@ def test_cpp_correct_overload_resolution():
def f(a):
return a

class A:
def __call__(self, a):
return a

assert m.dummy_function_overloaded_std_func_arg(f) == 9
assert m.dummy_function_overloaded_std_func_arg(A()) == 9

Check failure on line 115 in tests/test_callbacks.py

View workflow job for this annotation

GitHub Actions / 🐍 3.13 β€’ ubuntu-20.04 β€’ x64

test_cpp_correct_overload_resolution SystemError: <function test_cpp_correct_overload_resolution.<locals>.A.__call__ at 0x7f21a7725440> returned a result with an exception set

Check failure on line 115 in tests/test_callbacks.py

View workflow job for this annotation

GitHub Actions / 🐍 pypy-3.10 β€’ ubuntu-20.04 β€’ x64

test_cpp_correct_overload_resolution SystemError: An exception was set, but function returned a value

Check failure on line 115 in tests/test_callbacks.py

View workflow job for this annotation

GitHub Actions / 🐍 pypy-3.9 β€’ ubuntu-20.04 β€’ x64

test_cpp_correct_overload_resolution SystemError: An exception was set, but function returned a value

Check failure on line 115 in tests/test_callbacks.py

View workflow job for this annotation

GitHub Actions / 🐍 pypy-3.8 β€’ ubuntu-20.04 β€’ x64 -DPYBIND11_FINDPYTHON=ON

test_cpp_correct_overload_resolution SystemError: An exception was set, but function returned a value
assert m.dummy_function_overloaded_std_func_arg(lambda i: i) == 9

def f2(a, b):
return a + b

class B:
def __call__(self, a, b):
return a + b

assert m.dummy_function_overloaded_std_func_arg(f2) == 14
assert m.dummy_function_overloaded_std_func_arg(B()) == 14
assert m.dummy_function_overloaded_std_func_arg(lambda i, j: i + j) == 14


Expand Down

0 comments on commit 9768d5f

Please sign in to comment.