Commit 8b3f6190 authored by Kevin Modzelewski's avatar Kevin Modzelewski

Consolidate slot info into a slotdefs array

Previously had hardcoded three different iterations.
parent c4233571
...@@ -86,58 +86,68 @@ PyObject* Py_CallPythonRepr(PyObject* self) { ...@@ -86,58 +86,68 @@ PyObject* Py_CallPythonRepr(PyObject* self) {
} }
} }
bool update_slot(BoxedClass* self, const std::string& attr) { typedef wrapper_def slotdef;
if (attr == "__new__") {
self->tp_new = &Py_CallPythonNew;
// TODO update subclasses
return true;
}
if (attr == "__call__") { static void** slotptr(BoxedClass* self, int offset) {
self->tp_call = &Py_CallPythonCall; // TODO handle indices into the indirected portions (tp_as_sequence, etc)
// TODO update subclasses char* ptr = reinterpret_cast<char*>(self);
return reinterpret_cast<void**>(ptr + offset);
}
return true; static void update_one_slot(BoxedClass* self, const slotdef& p) {
} // TODO: CPython version is significantly more sophisticated
void** ptr = slotptr(self, p.offset);
assert(ptr);
if (attr == "__repr__") { if (typeLookup(self, p.name, NULL))
self->tp_repr = &Py_CallPythonRepr; *ptr = p.function;
// TODO update subclasses else
*ptr = NULL;
}
return true; static slotdef slotdefs[] = {
{ "__repr__", offsetof(PyTypeObject, tp_repr), (void*)&Py_CallPythonRepr, wrap_unaryfunc, 0 },
{ "__call__", offsetof(PyTypeObject, tp_call), (void*)&Py_CallPythonCall, (wrapperfunc)wrap_call,
PyWrapperFlag_KEYWORDS },
{ "__new__", offsetof(PyTypeObject, tp_new), (void*)&Py_CallPythonNew, NULL, 0 },
};
static void init_slotdefs() {
static bool initialized = false;
if (initialized)
return;
for (int i = 0; i < sizeof(slotdefs) / sizeof(slotdefs[0]); i++) {
if (i > 0) {
ASSERT(slotdefs[i].offset >= slotdefs[i - 1].offset, "%d %s", i, slotdefs[i - 1].name);
// CPython interns the name here
}
} }
return false; initialized = true;
} }
void fixup_slot_dispatchers(BoxedClass* self) { bool update_slot(BoxedClass* self, const std::string& attr) {
// This will probably share a lot in common with Py_TypeReady: bool updated = false;
if (!self->tp_new) { for (const slotdef& p : slotdefs) {
self->tp_new = &Py_CallPythonNew; if (p.name == attr) {
} else if (self->tp_new != Py_CallPythonNew) { // TODO update subclasses;
ASSERT(0, "need to set __new__?"); update_one_slot(self, p);
} updated = true;
if (!self->tp_call) {
self->tp_call = &Py_CallPythonCall;
} else if (self->tp_call != Py_CallPythonCall) {
ASSERT(0, "need to set __call__?");
} }
if (!self->tp_repr) {
self->tp_repr = &PyObject_Repr;
} else if (self->tp_repr != Py_CallPythonRepr) {
ASSERT(0, "need to set __repr__?");
} }
return updated;
} }
void fixup_slot_dispatchers(BoxedClass* self) {
init_slotdefs();
wrapper_def call_wrapper = { "__call__", offsetof(PyTypeObject, tp_call), (void*)&Py_CallPythonCall, for (const slotdef& p : slotdefs) {
(wrapperfunc)wrap_call, PyWrapperFlag_KEYWORDS }; update_one_slot(self, p);
wrapper_def repr_wrapper }
= { "__repr__", offsetof(PyTypeObject, tp_repr), (void*)&Py_CallPythonRepr, wrap_unaryfunc, 0 }; }
PyObject* tp_new_wrapper(PyTypeObject* self, BoxedTuple* args, Box* kwds) { static PyObject* tp_new_wrapper(PyTypeObject* self, BoxedTuple* args, Box* kwds) {
RELEASE_ASSERT(isSubclass(self->cls, type_cls), ""); RELEASE_ASSERT(isSubclass(self->cls, type_cls), "");
// ASSERT(self->tp_new != Py_CallPythonNew, "going to get in an infinite loop"); // ASSERT(self->tp_new != Py_CallPythonNew, "going to get in an infinite loop");
...@@ -155,20 +165,34 @@ PyObject* tp_new_wrapper(PyTypeObject* self, BoxedTuple* args, Box* kwds) { ...@@ -155,20 +165,34 @@ PyObject* tp_new_wrapper(PyTypeObject* self, BoxedTuple* args, Box* kwds) {
return self->tp_new(subtype, new_args, kwds); return self->tp_new(subtype, new_args, kwds);
} }
static void add_tp_new_wrapper(BoxedClass* type) {
if (type->getattr("__new__"))
return;
static void add_operators(PyTypeObject* cls) { type->giveAttr("__new__",
if (cls->tp_new) { new BoxedCApiFunction(METH_VARARGS | METH_KEYWORDS, type, "__new__", (PyCFunction)tp_new_wrapper));
cls->giveAttr("__new__", }
new BoxedCApiFunction(METH_VARARGS | METH_KEYWORDS, cls, "__new__", (PyCFunction)tp_new_wrapper));
}
if (cls->tp_call) { static void add_operators(BoxedClass* cls) {
cls->giveAttr("__call__", new BoxedWrapperDescriptor(&call_wrapper, cls)); init_slotdefs();
}
for (const slotdef& p : slotdefs) {
if (!p.wrapper)
continue;
if (cls->tp_repr) { void** ptr = slotptr(cls, p.offset);
cls->giveAttr("__repr__", new BoxedWrapperDescriptor(&repr_wrapper, cls));
if (!ptr || !*ptr)
continue;
if (cls->getattr(p.name))
continue;
// TODO PyObject_HashNotImplemented
cls->giveAttr(p.name, new BoxedWrapperDescriptor(&p, cls));
} }
if (cls->tp_new)
add_tp_new_wrapper(cls);
} }
extern "C" int PyType_IsSubtype(PyTypeObject* a, PyTypeObject* b) { extern "C" int PyType_IsSubtype(PyTypeObject* a, PyTypeObject* b) {
...@@ -243,7 +267,11 @@ extern "C" int PyType_Ready(PyTypeObject* cls) { ...@@ -243,7 +267,11 @@ extern "C" int PyType_Ready(PyTypeObject* cls) {
cls->tp_alloc = reinterpret_cast<decltype(cls->tp_alloc)>(PyType_GenericAlloc); cls->tp_alloc = reinterpret_cast<decltype(cls->tp_alloc)>(PyType_GenericAlloc);
} }
try {
add_operators(cls); add_operators(cls);
} catch (Box* b) {
abort();
}
for (PyMethodDef* method = cls->tp_methods; method && method->ml_name; ++method) { for (PyMethodDef* method = cls->tp_methods; method && method->ml_name; ++method) {
cls->giveAttr(method->ml_name, new BoxedMethodDescriptor(method, cls)); cls->giveAttr(method->ml_name, new BoxedMethodDescriptor(method, cls));
......
...@@ -19,7 +19,9 @@ ...@@ -19,7 +19,9 @@
namespace pyston { namespace pyston {
// Returns if a slot was updated
bool update_slot(BoxedClass* self, const std::string& attr); bool update_slot(BoxedClass* self, const std::string& attr);
void fixup_slot_dispatchers(BoxedClass* self); void fixup_slot_dispatchers(BoxedClass* self);
} }
......
...@@ -27,7 +27,9 @@ struct wrapper_def { ...@@ -27,7 +27,9 @@ struct wrapper_def {
int offset; int offset;
void* function; // "generic" handler that gets put in the tp_* slot which proxies to the python version void* function; // "generic" handler that gets put in the tp_* slot which proxies to the python version
wrapperfunc wrapper; // "wrapper" that ends up getting called by the Python-visible WrapperDescr wrapperfunc wrapper; // "wrapper" that ends up getting called by the Python-visible WrapperDescr
// exists in CPython: const char* doc
int flags; int flags;
// exists in CPython: PyObject *name_strobj
}; };
extern BoxedClass* capifunc_cls, *wrapperdescr_cls, *wrapperobject_cls; extern BoxedClass* capifunc_cls, *wrapperdescr_cls, *wrapperobject_cls;
......
...@@ -18,6 +18,7 @@ ...@@ -18,6 +18,7 @@
#include "llvm/Support/FileSystem.h" #include "llvm/Support/FileSystem.h"
#include "capi/typeobject.h"
#include "codegen/ast_interpreter.h" #include "codegen/ast_interpreter.h"
#include "codegen/irgen/hooks.h" #include "codegen/irgen/hooks.h"
#include "codegen/parser.h" #include "codegen/parser.h"
...@@ -528,6 +529,8 @@ extern "C" PyObject* PyErr_NewException(char* name, PyObject* _base, PyObject* d ...@@ -528,6 +529,8 @@ extern "C" PyObject* PyErr_NewException(char* name, PyObject* _base, PyObject* d
cls->giveAttr("__module__", boxStrConstantSize(name, dot_pos - name)); cls->giveAttr("__module__", boxStrConstantSize(name, dot_pos - name));
cls->giveAttr("__name__", boxStrConstantSize(dot_pos + 1, n - (dot_pos - name) - 1)); cls->giveAttr("__name__", boxStrConstantSize(dot_pos + 1, n - (dot_pos - name) - 1));
// TODO Not sure if this should be called here
fixup_slot_dispatchers(cls);
return cls; return cls;
} catch (Box* e) { } catch (Box* e) {
abort(); abort();
......
...@@ -3289,10 +3289,10 @@ Box* typeNew(Box* _cls, Box* arg1, Box* arg2, Box** _args) { ...@@ -3289,10 +3289,10 @@ Box* typeNew(Box* _cls, Box* arg1, Box* arg2, Box** _args) {
// Note: make sure to do this after assigning the attrs, since it will overwrite any defined __name__ // Note: make sure to do this after assigning the attrs, since it will overwrite any defined __name__
made->setattr("__name__", name, NULL); made->setattr("__name__", name, NULL);
// TODO this function (typeNew) should probably call PyType_Ready // TODO should this function (typeNew) call PyType_Ready?
made->tp_new = base->tp_new; made->tp_new = base->tp_new;
made->tp_alloc = reinterpret_cast<decltype(cls->tp_alloc)>(PyType_GenericAlloc); made->tp_alloc = reinterpret_cast<decltype(cls->tp_alloc)>(PyType_GenericAlloc);
fixup_slot_dispatchers(made);
return made; return made;
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment