mirror of
https://github.com/pybind/pybind11.git
synced 2025-02-16 13:47:53 +00:00
address issue with virtual function dispatch (fixes #159)
This commit is contained in:
parent
50ed3614de
commit
f5c154ade9
@ -9,7 +9,25 @@
|
|||||||
|
|
||||||
#include "example.h"
|
#include "example.h"
|
||||||
|
|
||||||
|
struct Base {
|
||||||
|
virtual void dispatch(void) const = 0;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct DispatchIssue : Base {
|
||||||
|
virtual void dispatch(void) const {
|
||||||
|
PYBIND11_OVERLOAD_PURE(void, Base, dispatch);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
void dispatch_issue_go(const Base * b) { b->dispatch(); }
|
||||||
|
|
||||||
|
PYBIND11_PLUGIN(mytest)
|
||||||
|
{
|
||||||
|
pybind11::module m("mytest", "A test");
|
||||||
|
|
||||||
|
|
||||||
|
return m.ptr();
|
||||||
|
}
|
||||||
void init_issues(py::module &m) {
|
void init_issues(py::module &m) {
|
||||||
py::module m2 = m.def_submodule("issues");
|
py::module m2 = m.def_submodule("issues");
|
||||||
|
|
||||||
@ -18,4 +36,12 @@ void init_issues(py::module &m) {
|
|||||||
|
|
||||||
// #150: char bindings broken
|
// #150: char bindings broken
|
||||||
m2.def("print_char", [](char c) { std::cout << c << std::endl; });
|
m2.def("print_char", [](char c) { std::cout << c << std::endl; });
|
||||||
|
|
||||||
|
// #159: virtual function dispatch has problems with similar-named functions
|
||||||
|
pybind11::class_<DispatchIssue> base(m2, "DispatchIssue");
|
||||||
|
base.alias<Base>()
|
||||||
|
.def(pybind11::init<>())
|
||||||
|
.def("dispatch", &Base::dispatch);
|
||||||
|
|
||||||
|
m2.def("dispatch_issue_go", &dispatch_issue_go);
|
||||||
}
|
}
|
||||||
|
@ -4,6 +4,25 @@ import sys
|
|||||||
sys.path.append('.')
|
sys.path.append('.')
|
||||||
|
|
||||||
from example.issues import print_cchar, print_char
|
from example.issues import print_cchar, print_char
|
||||||
|
from example.issues import DispatchIssue, dispatch_issue_go
|
||||||
|
|
||||||
print_cchar("const char *")
|
print_cchar("const char *")
|
||||||
print_char('c')
|
print_char('c')
|
||||||
|
|
||||||
|
|
||||||
|
class PyClass1(DispatchIssue):
|
||||||
|
def dispatch(self):
|
||||||
|
print("Yay..")
|
||||||
|
|
||||||
|
|
||||||
|
class PyClass2(DispatchIssue):
|
||||||
|
def dispatch(self):
|
||||||
|
try:
|
||||||
|
super(PyClass2, self).dispatch()
|
||||||
|
except Exception as e:
|
||||||
|
print("Failed as expected: " + str(e))
|
||||||
|
p = PyClass1()
|
||||||
|
dispatch_issue_go(p)
|
||||||
|
|
||||||
|
b = PyClass2()
|
||||||
|
dispatch_issue_go(b)
|
||||||
|
@ -1,2 +1,4 @@
|
|||||||
const char *
|
const char *
|
||||||
c
|
c
|
||||||
|
Failed as expected: Tried to call pure virtual function "dispatch"
|
||||||
|
Yay..
|
||||||
|
@ -1032,8 +1032,8 @@ inline function get_overload(const void *this_ptr, const char *name) {
|
|||||||
handle type = py_object.get_type();
|
handle type = py_object.get_type();
|
||||||
auto key = std::make_pair(type.ptr(), name);
|
auto key = std::make_pair(type.ptr(), name);
|
||||||
|
|
||||||
/* Cache functions that aren't overloaded in python to avoid
|
/* Cache functions that aren't overloaded in Python to avoid
|
||||||
many costly dictionary lookups in Python */
|
many costly Python dictionary lookups below */
|
||||||
auto &cache = detail::get_internals().inactive_overload_cache;
|
auto &cache = detail::get_internals().inactive_overload_cache;
|
||||||
if (cache.find(key) != cache.end())
|
if (cache.find(key) != cache.end())
|
||||||
return function();
|
return function();
|
||||||
@ -1044,10 +1044,16 @@ inline function get_overload(const void *this_ptr, const char *name) {
|
|||||||
return function();
|
return function();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Don't call dispatch code if invoked from overridden function */
|
||||||
PyFrameObject *frame = PyThreadState_Get()->frame;
|
PyFrameObject *frame = PyThreadState_Get()->frame;
|
||||||
pybind11::str caller = pybind11::handle(frame->f_code->co_name).str();
|
if ((std::string) pybind11::handle(frame->f_code->co_name).str() == name &&
|
||||||
if ((std::string) caller == name)
|
frame->f_code->co_argcount > 0) {
|
||||||
return function();
|
PyFrame_FastToLocals(frame);
|
||||||
|
PyObject *self_caller = PyDict_GetItem(
|
||||||
|
frame->f_locals, PyTuple_GET_ITEM(frame->f_code->co_varnames, 0));
|
||||||
|
if (self_caller == py_object.ptr())
|
||||||
|
return function();
|
||||||
|
}
|
||||||
return overload;
|
return overload;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user