Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
P
Pyston
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Boxiang Sun
Pyston
Commits
ad820078
Commit
ad820078
authored
Jan 08, 2015
by
Kevin Modzelewski
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Support nb_add
parent
60289a91
Changes
6
Hide whitespace changes
Inline
Side-by-side
Showing
6 changed files
with
206 additions
and
9 deletions
+206
-9
Makefile
Makefile
+1
-1
src/capi/object.cpp
src/capi/object.cpp
+25
-0
src/capi/typeobject.cpp
src/capi/typeobject.cpp
+155
-4
src/runtime/objmodel.cpp
src/runtime/objmodel.cpp
+1
-0
test/test_extension/slots_test.c
test/test_extension/slots_test.c
+19
-3
test/tests/capi_slots.py
test/tests/capi_slots.py
+5
-1
No files found.
Makefile
View file @
ad820078
...
@@ -321,7 +321,7 @@ NONSTDLIB_SRCS := $(MAIN_SRCS) $(OPTIONAL_SRCS) $(TOOL_SRCS) $(UNITTEST_SRCS)
...
@@ -321,7 +321,7 @@ NONSTDLIB_SRCS := $(MAIN_SRCS) $(OPTIONAL_SRCS) $(TOOL_SRCS) $(UNITTEST_SRCS)
# all: pyston_dbg pyston_release pyston_oprof pyston_prof $(OPTIONAL_SRCS:.cpp=.o) ext_python ext_pyston
# all: pyston_dbg pyston_release pyston_oprof pyston_prof $(OPTIONAL_SRCS:.cpp=.o) ext_python ext_pyston
all
:
pyston_dbg pyston_release pyston_prof ext_python ext_pyston unittests
all
:
pyston_dbg pyston_release pyston_prof ext_python ext_pyston unittests
ALL_HEADERS
:=
$(
wildcard
src/
*
/
*
.h
)
$(
wildcard
src/
*
/
*
/
*
.h
)
$(
wildcard
cpython2.7
/Include/
*
.h
)
ALL_HEADERS
:=
$(
wildcard
src/
*
/
*
.h
)
$(
wildcard
src/
*
/
*
/
*
.h
)
$(
wildcard
from_cpython
/Include/
*
.h
)
tags
:
$(SRCS) $(OPTIONAL_SRCS) $(FROM_CPYTHON_SRCS) $(ALL_HEADERS)
tags
:
$(SRCS) $(OPTIONAL_SRCS) $(FROM_CPYTHON_SRCS) $(ALL_HEADERS)
$(ECHO)
Calculating tags...
$(ECHO)
Calculating tags...
$(VERB)
ctags
$^
$(VERB)
ctags
$^
...
...
src/capi/object.cpp
View file @
ad820078
...
@@ -75,4 +75,29 @@ extern "C" int PyObject_GenericSetAttr(PyObject* obj, PyObject* name, PyObject*
...
@@ -75,4 +75,29 @@ extern "C" int PyObject_GenericSetAttr(PyObject* obj, PyObject* name, PyObject*
extern
"C"
int
PyObject_AsWriteBuffer
(
PyObject
*
obj
,
void
**
buffer
,
Py_ssize_t
*
buffer_len
)
{
extern
"C"
int
PyObject_AsWriteBuffer
(
PyObject
*
obj
,
void
**
buffer
,
Py_ssize_t
*
buffer_len
)
{
Py_FatalError
(
"unimplemented"
);
Py_FatalError
(
"unimplemented"
);
}
}
/* Return -1 if error; 1 if v op w; 0 if not (v op w). */
extern
"C"
int
PyObject_RichCompareBool
(
PyObject
*
v
,
PyObject
*
w
,
int
op
)
{
PyObject
*
res
;
int
ok
;
/* Quick result when objects are the same.
Guarantees that identity implies equality. */
if
(
v
==
w
)
{
if
(
op
==
Py_EQ
)
return
1
;
else
if
(
op
==
Py_NE
)
return
0
;
}
res
=
PyObject_RichCompare
(
v
,
w
,
op
);
if
(
res
==
NULL
)
return
-
1
;
if
(
PyBool_Check
(
res
))
ok
=
(
res
==
Py_True
);
else
ok
=
PyObject_IsTrue
(
res
);
Py_DECREF
(
res
);
return
ok
;
}
}
}
src/capi/typeobject.cpp
View file @
ad820078
...
@@ -109,6 +109,34 @@ static PyObject* wrap_binaryfunc(PyObject* self, PyObject* args, void* wrapped)
...
@@ -109,6 +109,34 @@ static PyObject* wrap_binaryfunc(PyObject* self, PyObject* args, void* wrapped)
return
(
*
func
)(
self
,
other
);
return
(
*
func
)(
self
,
other
);
}
}
static
PyObject
*
wrap_binaryfunc_l
(
PyObject
*
self
,
PyObject
*
args
,
void
*
wrapped
)
{
binaryfunc
func
=
(
binaryfunc
)
wrapped
;
PyObject
*
other
;
if
(
!
check_num_args
(
args
,
1
))
return
NULL
;
other
=
PyTuple_GET_ITEM
(
args
,
0
);
if
(
!
(
self
->
cls
->
tp_flags
&
Py_TPFLAGS_CHECKTYPES
)
&&
!
PyType_IsSubtype
(
other
->
cls
,
self
->
cls
))
{
Py_INCREF
(
Py_NotImplemented
);
return
Py_NotImplemented
;
}
return
(
*
func
)(
self
,
other
);
}
static
PyObject
*
wrap_binaryfunc_r
(
PyObject
*
self
,
PyObject
*
args
,
void
*
wrapped
)
{
binaryfunc
func
=
(
binaryfunc
)
wrapped
;
PyObject
*
other
;
if
(
!
check_num_args
(
args
,
1
))
return
NULL
;
other
=
PyTuple_GET_ITEM
(
args
,
0
);
if
(
!
(
self
->
cls
->
tp_flags
&
Py_TPFLAGS_CHECKTYPES
)
&&
!
PyType_IsSubtype
(
other
->
cls
,
self
->
cls
))
{
Py_INCREF
(
Py_NotImplemented
);
return
Py_NotImplemented
;
}
return
(
*
func
)(
other
,
self
);
}
static
Py_ssize_t
getindex
(
PyObject
*
self
,
PyObject
*
arg
)
noexcept
{
static
Py_ssize_t
getindex
(
PyObject
*
self
,
PyObject
*
arg
)
noexcept
{
Py_ssize_t
i
;
Py_ssize_t
i
;
...
@@ -355,6 +383,41 @@ static PyObject* call_method(PyObject* o, const char* name, PyObject** nameobj,
...
@@ -355,6 +383,41 @@ static PyObject* call_method(PyObject* o, const char* name, PyObject** nameobj,
return
retval
;
return
retval
;
}
}
/* Clone of call_method() that returns NotImplemented when the lookup fails. */
static
PyObject
*
call_maybe
(
PyObject
*
o
,
const
char
*
name
,
PyObject
**
nameobj
,
const
char
*
format
,
...)
noexcept
{
va_list
va
;
PyObject
*
args
,
*
func
=
0
,
*
retval
;
va_start
(
va
,
format
);
func
=
lookup_maybe
(
o
,
name
,
nameobj
);
if
(
func
==
NULL
)
{
va_end
(
va
);
if
(
!
PyErr_Occurred
())
{
Py_INCREF
(
Py_NotImplemented
);
return
Py_NotImplemented
;
}
return
NULL
;
}
if
(
format
&&
*
format
)
args
=
Py_VaBuildValue
(
format
,
va
);
else
args
=
PyTuple_New
(
0
);
va_end
(
va
);
if
(
args
==
NULL
)
return
NULL
;
assert
(
PyTuple_Check
(
args
));
retval
=
PyObject_Call
(
func
,
args
,
NULL
);
Py_DECREF
(
args
);
Py_DECREF
(
func
);
return
retval
;
}
PyObject
*
slot_tp_repr
(
PyObject
*
self
)
noexcept
{
PyObject
*
slot_tp_repr
(
PyObject
*
self
)
noexcept
{
try
{
try
{
...
@@ -646,6 +709,73 @@ static int slot_sq_contains(PyObject* self, PyObject* value) {
...
@@ -646,6 +709,73 @@ static int slot_sq_contains(PyObject* self, PyObject* value) {
return call_method(self, OPSTR, &cache_str, "(" ARGCODES ")", arg1); \
return call_method(self, OPSTR, &cache_str, "(" ARGCODES ")", arg1); \
}
}
/* Boolean helper for SLOT1BINFULL().
right.__class__ is a nontrivial subclass of left.__class__. */
static
int
method_is_overloaded
(
PyObject
*
left
,
PyObject
*
right
,
const
char
*
name
)
{
PyObject
*
a
,
*
b
;
int
ok
;
b
=
PyObject_GetAttrString
((
PyObject
*
)(
Py_TYPE
(
right
)),
name
);
if
(
b
==
NULL
)
{
PyErr_Clear
();
/* If right doesn't have it, it's not overloaded */
return
0
;
}
a
=
PyObject_GetAttrString
((
PyObject
*
)(
Py_TYPE
(
left
)),
name
);
if
(
a
==
NULL
)
{
PyErr_Clear
();
Py_DECREF
(
b
);
/* If right has it but left doesn't, it's overloaded */
return
1
;
}
ok
=
PyObject_RichCompareBool
(
a
,
b
,
Py_NE
);
Py_DECREF
(
a
);
Py_DECREF
(
b
);
if
(
ok
<
0
)
{
PyErr_Clear
();
return
0
;
}
return
ok
;
}
#define SLOT1BINFULL(FUNCNAME, TESTFUNC, SLOTNAME, OPSTR, ROPSTR) \
static PyObject* FUNCNAME(PyObject* self, PyObject* other) { \
static PyObject* cache_str, *rcache_str; \
int do_other = Py_TYPE(self) != Py_TYPE(other) && Py_TYPE(other)->tp_as_number != NULL \
&& Py_TYPE(other)->tp_as_number->SLOTNAME == TESTFUNC; \
if (Py_TYPE(self)->tp_as_number != NULL && Py_TYPE(self)->tp_as_number->SLOTNAME == TESTFUNC) { \
PyObject* r; \
if (do_other && PyType_IsSubtype(Py_TYPE(other), Py_TYPE(self)) \
&& method_is_overloaded(self, other, ROPSTR)) { \
r = call_maybe(other, ROPSTR, &rcache_str, "(O)", self); \
if (r != Py_NotImplemented) \
return r; \
Py_DECREF(r); \
do_other = 0; \
} \
r = call_maybe(self, OPSTR, &cache_str, "(O)", other); \
if (r != Py_NotImplemented || Py_TYPE(other) == Py_TYPE(self)) \
return r; \
Py_DECREF(r); \
} \
if (do_other) { \
return call_maybe(other, ROPSTR, &rcache_str, "(O)", self); \
} \
Py_INCREF(Py_NotImplemented); \
return Py_NotImplemented; \
}
#define SLOT1BIN(FUNCNAME, SLOTNAME, OPSTR, ROPSTR) SLOT1BINFULL(FUNCNAME, FUNCNAME, SLOTNAME, OPSTR, ROPSTR)
#define SLOT2(FUNCNAME, OPSTR, ARG1TYPE, ARG2TYPE, ARGCODES) \
static PyObject* FUNCNAME(PyObject* self, ARG1TYPE arg1, ARG2TYPE arg2) { \
static PyObject* cache_str; \
return call_method(self, OPSTR, &cache_str, "(" ARGCODES ")", arg1, arg2); \
}
#define slot_mp_length slot_sq_length
#define slot_mp_length slot_sq_length
SLOT1
(
slot_mp_subscript
,
"__getitem__"
,
PyObject
*
,
"O"
)
SLOT1
(
slot_mp_subscript
,
"__getitem__"
,
PyObject
*
,
"O"
)
...
@@ -664,6 +794,14 @@ static int slot_mp_ass_subscript(PyObject* self, PyObject* key, PyObject* value)
...
@@ -664,6 +794,14 @@ static int slot_mp_ass_subscript(PyObject* self, PyObject* key, PyObject* value)
return
0
;
return
0
;
}
}
SLOT1BIN
(
slot_nb_add
,
nb_add
,
"__add__"
,
"__radd__"
)
SLOT1BIN
(
slot_nb_subtract
,
nb_subtract
,
"__sub__"
,
"__rsub__"
)
SLOT1BIN
(
slot_nb_multiply
,
nb_multiply
,
"__mul__"
,
"__rmul__"
)
SLOT1BIN
(
slot_nb_divide
,
nb_divide
,
"__div__"
,
"__rdiv__"
)
SLOT1BIN
(
slot_nb_remainder
,
nb_remainder
,
"__mod__"
,
"__rmod__"
)
SLOT1BIN
(
slot_nb_divmod
,
nb_divmod
,
"__divmod__"
,
"__rdivmod__"
)
static
PyObject
*
slot_nb_power
(
PyObject
*
,
PyObject
*
,
PyObject
*
);
typedef
wrapper_def
slotdef
;
typedef
wrapper_def
slotdef
;
...
@@ -755,6 +893,8 @@ static slotdef slotdefs[] = {
...
@@ -755,6 +893,8 @@ static slotdef slotdefs[] = {
PyWrapperFlag_KEYWORDS
),
PyWrapperFlag_KEYWORDS
),
TPSLOT
(
"__new__"
,
tp_new
,
slot_tp_new
,
NULL
,
""
),
TPSLOT
(
"__new__"
,
tp_new
,
slot_tp_new
,
NULL
,
""
),
BINSLOT
(
"__add__"
,
nb_add
,
slot_nb_add
,
"+"
),
//
RBINSLOT
(
"__radd__"
,
nb_add
,
slot_nb_add
,
"+"
),
UNSLOT
(
"__nonzero__"
,
nb_nonzero
,
slot_nb_nonzero
,
wrap_inquirypred
,
"x != 0"
),
UNSLOT
(
"__nonzero__"
,
nb_nonzero
,
slot_nb_nonzero
,
wrap_inquirypred
,
"x != 0"
),
MPSLOT
(
"__len__"
,
mp_length
,
slot_mp_length
,
wrap_lenfunc
,
"x.__len__() <==> len(x)"
),
MPSLOT
(
"__len__"
,
mp_length
,
slot_mp_length
,
wrap_lenfunc
,
"x.__len__() <==> len(x)"
),
...
@@ -913,22 +1053,33 @@ extern "C" int PyType_Ready(PyTypeObject* cls) {
...
@@ -913,22 +1053,33 @@ extern "C" int PyType_Ready(PyTypeObject* cls) {
// zero out the ones we know about, then assert that the entire struct
// zero out the ones we know about, then assert that the entire struct
// is zero, then restore the ones we know about.
// is zero, then restore the ones we know about.
if
(
cls
->
tp_as_number
)
{
if
(
cls
->
tp_as_number
)
{
auto
nb_nonzero
=
cls
->
tp_as_number
->
nb_nonzero
;
#define SAVE(N) \
cls
->
tp_as_number
->
nb_nonzero
=
NULL
;
auto N = cls->tp_as_number->N; \
cls->tp_as_number->N = NULL;
#define RESTORE(N) cls->tp_as_number->N = N;
SAVE
(
nb_nonzero
);
SAVE
(
nb_add
);
for
(
void
**
p
=
(
void
**
)
cls
->
tp_as_number
;
p
<
(
void
**
)
cls
->
tp_as_number
+
1
;
p
++
)
{
for
(
void
**
p
=
(
void
**
)
cls
->
tp_as_number
;
p
<
(
void
**
)
cls
->
tp_as_number
+
1
;
p
++
)
{
RELEASE_ASSERT
(
*
p
==
NULL
,
""
);
RELEASE_ASSERT
(
*
p
==
NULL
,
""
);
}
}
cls
->
tp_as_number
->
nb_nonzero
=
nb_nonzero
;
RESTORE
(
nb_nonzero
);
RESTORE
(
nb_add
)
#undef SAVE
#undef RESTORE
}
}
RELEASE_ASSERT
(
cls
->
tp_getattro
==
NULL
||
cls
->
tp_getattro
==
PyObject_GenericGetAttr
,
""
);
RELEASE_ASSERT
(
cls
->
tp_getattro
==
NULL
||
cls
->
tp_getattro
==
PyObject_GenericGetAttr
,
""
);
RELEASE_ASSERT
(
cls
->
tp_setattro
==
NULL
||
cls
->
tp_setattro
==
PyObject_GenericSetAttr
,
""
);
RELEASE_ASSERT
(
cls
->
tp_setattro
==
NULL
||
cls
->
tp_setattro
==
PyObject_GenericSetAttr
,
""
);
RELEASE_ASSERT
(
cls
->
tp_as_buffer
==
NULL
,
""
);
RELEASE_ASSERT
(
cls
->
tp_as_buffer
==
NULL
,
""
);
int
ALLOWABLE_FLAGS
=
Py_TPFLAGS_DEFAULT
|
Py_TPFLAGS_BASETYPE
|
Py_TPFLAGS_HAVE_GC
;
int
ALLOWABLE_FLAGS
=
Py_TPFLAGS_DEFAULT
|
Py_TPFLAGS_BASETYPE
|
Py_TPFLAGS_HAVE_GC
|
Py_TPFLAGS_CHECKTYPES
;
RELEASE_ASSERT
((
cls
->
tp_flags
&
~
ALLOWABLE_FLAGS
)
==
0
,
""
);
RELEASE_ASSERT
((
cls
->
tp_flags
&
~
ALLOWABLE_FLAGS
)
==
0
,
""
);
if
(
cls
->
tp_as_number
)
{
RELEASE_ASSERT
(
cls
->
tp_flags
&
Py_TPFLAGS_CHECKTYPES
,
"Pyston doesn't yet support non-checktypes behavior"
);
}
RELEASE_ASSERT
(
cls
->
tp_iter
==
NULL
,
""
);
RELEASE_ASSERT
(
cls
->
tp_iter
==
NULL
,
""
);
RELEASE_ASSERT
(
cls
->
tp_iternext
==
NULL
,
""
);
RELEASE_ASSERT
(
cls
->
tp_iternext
==
NULL
,
""
);
...
...
src/runtime/objmodel.cpp
View file @
ad820078
...
@@ -353,6 +353,7 @@ BoxedClass::BoxedClass(BoxedClass* metaclass, BoxedClass* base, gcvisit_func gc_
...
@@ -353,6 +353,7 @@ BoxedClass::BoxedClass(BoxedClass* metaclass, BoxedClass* base, gcvisit_func gc_
tp_basicsize
=
instance_size
;
tp_basicsize
=
instance_size
;
tp_flags
|=
Py_TPFLAGS_HEAPTYPE
;
tp_flags
|=
Py_TPFLAGS_HEAPTYPE
;
tp_flags
|=
Py_TPFLAGS_CHECKTYPES
;
if
(
metaclass
==
NULL
)
{
if
(
metaclass
==
NULL
)
{
assert
(
type_cls
==
NULL
);
assert
(
type_cls
==
NULL
);
...
...
test/test_extension/slots_test.c
View file @
ad820078
...
@@ -215,8 +215,14 @@ static int s_nonzero(slots_tester_object* self) {
...
@@ -215,8 +215,14 @@ static int s_nonzero(slots_tester_object* self) {
return
self
->
n
!=
0
;
return
self
->
n
!=
0
;
}
}
static
PyObject
*
s_add
(
slots_tester_object
*
lhs
,
PyObject
*
rhs
)
{
printf
(
"s_add, %d %s
\n
"
,
lhs
->
n
,
Py_TYPE
(
rhs
)
->
tp_name
);
Py_INCREF
(
lhs
);
return
(
PyObject
*
)
lhs
;
}
static
PyNumberMethods
slots_tester_as_number
=
{
static
PyNumberMethods
slots_tester_as_number
=
{
0
,
/* nb_add */
(
binaryfunc
)
s_add
,
/* nb_add */
0
,
/* nb_subtract */
0
,
/* nb_subtract */
0
,
/* nb_multiply */
0
,
/* nb_multiply */
0
,
/* nb_divide */
0
,
/* nb_divide */
...
@@ -273,11 +279,11 @@ static PyTypeObject slots_tester_num = {
...
@@ -273,11 +279,11 @@ static PyTypeObject slots_tester_num = {
0
,
/* tp_as_mapping */
0
,
/* tp_as_mapping */
0
,
/* tp_hash */
0
,
/* tp_hash */
0
,
/* tp_call */
0
,
/* tp_call */
0
,
/* tp_str */
(
reprfunc
)
slots_tester_seq_str
,
/* tp_str */
0
,
/* tp_getattro */
0
,
/* tp_getattro */
0
,
/* tp_setattro */
0
,
/* tp_setattro */
0
,
/* tp_as_buffer */
0
,
/* tp_as_buffer */
Py_TPFLAGS_DEFAULT
,
/* tp_flags */
Py_TPFLAGS_DEFAULT
|
Py_TPFLAGS_CHECKTYPES
,
/* tp_flags */
0
,
/* tp_doc */
0
,
/* tp_doc */
0
,
/* tp_traverse */
0
,
/* tp_traverse */
0
,
/* tp_clear */
0
,
/* tp_clear */
...
@@ -383,10 +389,20 @@ call_funcs(PyObject* _module, PyObject* args) {
...
@@ -383,10 +389,20 @@ call_funcs(PyObject* _module, PyObject* args) {
printf
(
"tp_as_number exists
\n
"
);
printf
(
"tp_as_number exists
\n
"
);
PyNumberMethods
*
num
=
cls
->
tp_as_number
;
PyNumberMethods
*
num
=
cls
->
tp_as_number
;
if
(
!
(
cls
->
tp_flags
&
Py_TPFLAGS_CHECKTYPES
))
{
printf
(
"CHECKTYPES is not set!
\n
"
);
}
if
(
num
->
nb_nonzero
)
{
if
(
num
->
nb_nonzero
)
{
int
n
=
num
->
nb_nonzero
(
obj
);
int
n
=
num
->
nb_nonzero
(
obj
);
printf
(
"nb_nonzero exists and returned %d
\n
"
,
n
);
printf
(
"nb_nonzero exists and returned %d
\n
"
,
n
);
}
}
if
(
num
->
nb_add
)
{
PyObject
*
res
=
num
->
nb_add
(
obj
,
obj
);
printf
(
"nb_add exists and returned a %s
\n
"
,
Py_TYPE
(
res
)
->
tp_name
);
Py_DECREF
(
res
);
}
}
else
{
}
else
{
printf
(
"tp_as_number doesnt exist
\n
"
);
printf
(
"tp_as_number doesnt exist
\n
"
);
}
}
...
...
test/tests/capi_slots.py
View file @
ad820078
...
@@ -16,7 +16,7 @@ for i in xrange(3):
...
@@ -16,7 +16,7 @@ for i in xrange(3):
for
i
in
xrange
(
3
):
for
i
in
xrange
(
3
):
t
=
slots_test
.
SlotsTesterNum
(
i
)
t
=
slots_test
.
SlotsTesterNum
(
i
)
print
bool
(
t
)
print
bool
(
t
)
,
t
+
5
class
C
(
object
):
class
C
(
object
):
def
__repr__
(
self
):
def
__repr__
(
self
):
...
@@ -43,4 +43,8 @@ def nonzero(self):
...
@@ -43,4 +43,8 @@ def nonzero(self):
return
True
return
True
C
.
__nonzero__
=
nonzero
C
.
__nonzero__
=
nonzero
def
add
(
self
,
rhs
):
print
"add"
,
self
,
rhs
C
.
__add__
=
add
slots_test
.
call_funcs
(
C
())
slots_test
.
call_funcs
(
C
())
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment