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
0b1acea8
Commit
0b1acea8
authored
Feb 11, 2016
by
Kevin Modzelewski
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Working on getting old-style classes working
parent
3874fd6c
Changes
11
Hide whitespace changes
Inline
Side-by-side
Showing
11 changed files
with
184 additions
and
70 deletions
+184
-70
src/capi/modsupport.cpp
src/capi/modsupport.cpp
+2
-2
src/capi/typeobject.cpp
src/capi/typeobject.cpp
+1
-1
src/codegen/ast_interpreter.cpp
src/codegen/ast_interpreter.cpp
+1
-1
src/codegen/irgen/hooks.cpp
src/codegen/irgen/hooks.cpp
+7
-3
src/codegen/unwinding.cpp
src/codegen/unwinding.cpp
+4
-3
src/core/types.h
src/core/types.h
+3
-2
src/runtime/builtin_modules/builtins.cpp
src/runtime/builtin_modules/builtins.cpp
+1
-1
src/runtime/classobj.cpp
src/runtime/classobj.cpp
+141
-26
src/runtime/classobj.h
src/runtime/classobj.h
+14
-21
src/runtime/objmodel.cpp
src/runtime/objmodel.cpp
+2
-3
src/runtime/types.cpp
src/runtime/types.cpp
+8
-7
No files found.
src/capi/modsupport.cpp
View file @
0b1acea8
...
...
@@ -431,11 +431,11 @@ extern "C" PyObject* Py_InitModule4(const char* name, PyMethodDef* methods, cons
return
module
;
}
extern
"C"
PyObject
*
PyModule_GetDict
(
PyObject
*
_m
)
noexcept
{
extern
"C"
PyObject
*
PyModule_GetDict
(
BORROWED
(
PyObject
*
)
_m
)
noexcept
{
BoxedModule
*
m
=
static_cast
<
BoxedModule
*>
(
_m
);
assert
(
m
->
cls
==
module_cls
);
return
m
->
getAttrWrapper
(
);
return
autoDecref
(
m
->
getAttrWrapper
()
);
}
extern
"C"
int
PyModule_AddObject
(
PyObject
*
_m
,
const
char
*
name
,
PyObject
*
value
)
noexcept
{
...
...
src/capi/typeobject.cpp
View file @
0b1acea8
...
...
@@ -3427,7 +3427,7 @@ extern "C" int PyType_Ready(PyTypeObject* cls) noexcept {
cls
->
giveAttrBorrowed
(
"__base__"
,
base
);
assert
(
cls
->
tp_dict
==
NULL
);
cls
->
tp_dict
=
incref
(
cls
->
getAttrWrapper
()
);
cls
->
tp_dict
=
cls
->
getAttrWrapper
(
);
assert
(
cls
->
tp_name
);
...
...
src/codegen/ast_interpreter.cpp
View file @
0b1acea8
...
...
@@ -184,7 +184,7 @@ public:
Box
*
getGlobals
()
{
assert
(
globals
);
return
globals
;
return
incref
(
globals
)
;
}
FunctionMetadata
*
getMD
()
{
return
md
;
}
...
...
src/codegen/irgen/hooks.cpp
View file @
0b1acea8
...
...
@@ -331,7 +331,7 @@ void compileAndRunModule(AST_Module* m, BoxedModule* bm) {
static
BoxedString
*
builtins_str
=
getStaticString
(
"__builtins__"
);
if
(
!
bm
->
hasattr
(
builtins_str
))
bm
->
giveAttr
(
builtins_str
,
PyModule_GetDict
(
builtins_module
));
bm
->
giveAttr
(
incref
(
builtins_str
)
,
PyModule_GetDict
(
builtins_module
));
md
=
new
FunctionMetadata
(
0
,
false
,
false
,
std
::
move
(
si
));
}
...
...
@@ -565,16 +565,20 @@ static void pickGlobalsAndLocals(Box*& globals, Box*& locals) {
if
(
globals
)
{
// From CPython (they set it to be f->f_builtins):
Box
*
globals_dict
=
globals
;
Box
*
globals_dict
;
if
(
globals
->
cls
==
module_cls
)
globals_dict
=
globals
->
getAttrWrapper
();
else
globals_dict
=
incref
(
globals
);
AUTO_DECREF
(
globals_dict
);
auto
requested_builtins
=
PyDict_GetItemString
(
globals_dict
,
"__builtins__"
);
if
(
requested_builtins
==
NULL
)
PyDict_SetItemString
(
globals_dict
,
"__builtins__"
,
builtins_module
);
else
RELEASE_ASSERT
(
requested_builtins
==
builtins_module
||
requested_builtins
==
builtins_module
->
getAttrWrapper
(
),
||
requested_builtins
==
autoDecref
(
builtins_module
->
getAttrWrapper
()
),
"we don't support overriding __builtins__"
);
}
}
...
...
src/codegen/unwinding.cpp
View file @
0b1acea8
...
...
@@ -369,7 +369,7 @@ public:
auto
locations
=
findLocations
(
PASSED_GLOBALS_NAME
);
assert
(
locations
.
size
()
==
1
);
Box
*
r
=
(
Box
*
)
readLocation
(
locations
[
0
]);
return
r
;
return
incref
(
r
)
;
}
else
if
(
id
.
type
==
PythonFrameId
::
INTERPRETED
)
{
return
getGlobalsForInterpretedFrame
((
void
*
)
id
.
bp
);
}
...
...
@@ -381,8 +381,10 @@ public:
if
(
!
globals
)
return
NULL
;
if
(
PyModule_Check
(
globals
))
if
(
PyModule_Check
(
globals
))
{
AUTO_DECREF
(
globals
);
return
globals
->
getAttrWrapper
();
}
return
globals
;
}
...
...
@@ -785,7 +787,6 @@ Box* getGlobals() {
auto
it
=
getTopPythonFrame
();
if
(
!
it
)
return
NULL
;
RELEASE_ASSERT
(
0
,
"check refcounting"
);
return
it
->
getGlobals
();
}
...
...
src/core/types.h
View file @
0b1acea8
...
...
@@ -670,7 +670,7 @@ public:
Py_INCREF
(
val
);
giveAttr
(
internStringMortal
(
attr
),
val
);
}
void
giveAttr
(
BoxedString
*
attr
,
Box
*
val
);
void
giveAttr
(
STOLEN
(
BoxedString
*
)
attr
,
STOLEN
(
Box
*
)
val
);
void
clearAttrs
();
...
...
@@ -685,7 +685,8 @@ public:
void
delattr
(
BoxedString
*
attr
,
DelattrRewriteArgs
*
rewrite_args
);
// Only valid for hc-backed instances:
BORROWED
(
Box
*
)
getAttrWrapper
();
// Maybe this should return a borrowed reference?
Box
*
getAttrWrapper
();
Box
*
reprIC
();
BoxedString
*
reprICAsString
();
...
...
src/runtime/builtin_modules/builtins.cpp
View file @
0b1acea8
...
...
@@ -1067,7 +1067,7 @@ public:
static
Box
*
__reduce__
(
Box
*
self
)
{
RELEASE_ASSERT
(
isSubclass
(
self
->
cls
,
BaseException
),
""
);
BoxedException
*
exc
=
static_cast
<
BoxedException
*>
(
self
);
return
BoxedTuple
::
create
({
self
->
cls
,
EmptyTuple
,
self
->
getAttrWrapper
(
)
});
return
BoxedTuple
::
create
({
self
->
cls
,
EmptyTuple
,
autoDecref
(
self
->
getAttrWrapper
()
)
});
}
};
...
...
src/runtime/classobj.cpp
View file @
0b1acea8
...
...
@@ -31,7 +31,7 @@ BoxedClass* classobj_cls, *instance_cls;
}
template
<
Rewritable
rewritable
>
static
B
ox
*
classLookup
(
BoxedClassobj
*
cls
,
BoxedString
*
attr
,
GetattrRewriteArgs
*
rewrite_args
)
{
static
B
ORROWED
(
Box
*
)
classLookup
(
BoxedClassobj
*
cls
,
BoxedString
*
attr
,
GetattrRewriteArgs
*
rewrite_args
)
{
if
(
rewritable
==
NOT_REWRITABLE
)
{
assert
(
!
rewrite_args
);
rewrite_args
=
NULL
;
...
...
@@ -162,7 +162,7 @@ Box* classobjNew(Box* _cls, Box* _name, Box* _bases, Box** _args) {
BoxedClassobj
*
made
=
new
(
cls
)
BoxedClassobj
(
name
,
bases
);
made
->
giveAttr
(
"__module__"
,
boxString
(
getCurrentModule
()
->
name
()));
made
->
giveAttr
(
"__doc__"
,
None
);
made
->
giveAttr
(
"__doc__"
,
incref
(
None
)
);
for
(
const
auto
&
p
:
*
dict
)
{
RELEASE_ASSERT
(
p
.
first
->
cls
==
str_cls
,
""
);
...
...
@@ -184,13 +184,16 @@ Box* classobjCall(Box* _cls, Box* _args, Box* _kwargs) {
assert
(
!
_kwargs
||
_kwargs
->
cls
==
dict_cls
);
BoxedDict
*
kwargs
=
static_cast
<
BoxedDict
*>
(
_kwargs
);
BoxedInstance
*
made
=
new
BoxedInstance
(
cls
);
static
BoxedString
*
init_str
=
getStaticString
(
"__init__"
);
Box
*
init_func
=
classLookup
(
cls
,
init_str
);
BoxedInstance
*
made
=
new
BoxedInstance
(
cls
);
Py_DECREF
(
made
);
Py_RETURN_NONE
;
if
(
init_func
)
{
Box
*
init_rtn
=
runtimeCall
(
init_func
,
ArgPassSpec
(
1
,
0
,
true
,
true
),
made
,
args
,
kwargs
,
NULL
,
NULL
);
AUTO_DECREF
(
init_rtn
);
if
(
init_rtn
!=
None
)
raiseExcHelper
(
TypeError
,
"__init__() should return None"
);
}
else
{
...
...
@@ -210,6 +213,7 @@ extern "C" PyObject* PyInstance_New(PyObject* klass, PyObject* arg, PyObject* kw
}
static
Box
*
classobjGetattribute
(
Box
*
_cls
,
Box
*
_attr
)
{
assert
(
0
&&
"check refcounting"
);
RELEASE_ASSERT
(
_cls
->
cls
==
classobj_cls
,
""
);
BoxedClassobj
*
cls
=
static_cast
<
BoxedClassobj
*>
(
_cls
);
...
...
@@ -379,6 +383,7 @@ static Box* instanceGetattributeSimple(BoxedInstance* inst, BoxedString* attr_st
rewrite_args
=
NULL
;
if
(
r
)
{
assert
(
0
&&
"erf this was supposed to return a borrowed ref"
);
Box
*
rtn
=
processDescriptor
(
r
,
inst
,
inst
->
inst_cls
);
if
(
rewrite_args
)
{
RewriterVar
*
r_rtn
=
rewrite_args
->
rewriter
->
call
(
...
...
@@ -439,6 +444,7 @@ static Box* instanceGetattributeWithFallback(BoxedInstance* inst, BoxedString* a
template
<
Rewritable
rewritable
>
static
Box
*
_instanceGetattribute
(
Box
*
_inst
,
BoxedString
*
attr_str
,
bool
raise_on_missing
,
GetattrRewriteArgs
*
rewrite_args
)
{
assert
(
0
&&
"check refcounting"
);
if
(
rewritable
==
NOT_REWRITABLE
)
{
assert
(
!
rewrite_args
);
rewrite_args
=
NULL
;
...
...
@@ -498,7 +504,7 @@ Box* instanceGetattroInternal(Box* cls, Box* _attr, GetattrRewriteArgs* rewrite_
template
Box
*
instanceGetattroInternal
<
CAPI
>(
Box
*
,
Box
*
,
GetattrRewriteArgs
*
)
noexcept
;
template
Box
*
instanceGetattroInternal
<
CXX
>(
Box
*
,
Box
*
,
GetattrRewriteArgs
*
);
Box
*
instanceSetattroInternal
(
Box
*
_inst
,
Box
*
_attr
,
Box
*
value
,
SetattrRewriteArgs
*
rewrite_args
)
{
void
instanceSetattroInternal
(
Box
*
_inst
,
Box
*
_attr
,
STOLEN
(
Box
*
)
value
,
SetattrRewriteArgs
*
rewrite_args
)
{
STAT_TIMER
(
t0
,
"us_timer_instance_setattro"
,
0
);
RELEASE_ASSERT
(
_inst
->
cls
==
instance_cls
,
""
);
...
...
@@ -519,7 +525,7 @@ Box* instanceSetattroInternal(Box* _inst, Box* _attr, Box* value, SetattrRewrite
raiseExcHelper
(
TypeError
,
"__class__ must be set to a class"
);
inst
->
inst_cls
=
static_cast
<
BoxedClassobj
*>
(
value
);
return
None
;
return
;
}
}
...
...
@@ -539,28 +545,30 @@ Box* instanceSetattroInternal(Box* _inst, Box* _attr, Box* value, SetattrRewrite
if
(
setattr
)
{
setattr
=
processDescriptor
(
setattr
,
inst
,
inst
->
inst_cls
);
return
runtimeCall
(
setattr
,
ArgPassSpec
(
2
),
_attr
,
value
,
NULL
,
NULL
,
NULL
);
runtimeCall
(
setattr
,
ArgPassSpec
(
2
),
_attr
,
value
,
NULL
,
NULL
,
NULL
);
return
;
}
if
(
rewrite_args
)
grewrite_args
.
assertReturnConvention
(
ReturnConvention
::
NO_RETURN
);
_inst
->
setattr
(
attr
,
value
,
rewrite_args
);
return
None
;
return
;
}
else
{
Box
*
setattr
=
classLookup
(
inst
->
inst_cls
,
setattr_str
);
if
(
setattr
)
{
setattr
=
processDescriptor
(
setattr
,
inst
,
inst
->
inst_cls
);
return
runtimeCall
(
setattr
,
ArgPassSpec
(
2
),
_attr
,
value
,
NULL
,
NULL
,
NULL
);
runtimeCall
(
setattr
,
ArgPassSpec
(
2
),
_attr
,
value
,
NULL
,
NULL
,
NULL
);
return
;
}
_inst
->
setattr
(
attr
,
value
,
NULL
);
return
None
;
return
;
}
}
Box
*
instanceSetattr
(
Box
*
_inst
,
Box
*
_attr
,
Box
*
value
)
{
return
instanceSetattroInternal
(
_inst
,
_attr
,
value
,
NULL
);
void
instanceSetattr
(
Box
*
_inst
,
Box
*
_attr
,
Box
*
value
)
{
instanceSetattroInternal
(
_inst
,
_attr
,
value
,
NULL
);
}
Box
*
instanceDelattr
(
Box
*
_inst
,
Box
*
_attr
)
{
...
...
@@ -1198,20 +1206,6 @@ static PyObject* instance_index(PyObject* self) noexcept {
return
res
;
}
static
void
instance_dealloc
(
Box
*
_inst
)
noexcept
{
RELEASE_ASSERT
(
_inst
->
cls
==
instance_cls
,
""
);
BoxedInstance
*
inst
=
static_cast
<
BoxedInstance
*>
(
_inst
);
// Note that trying to call __del__ as a finalizer does not fallback to
// __getattr__ unlike other attributes (like __index__). This is CPython's behavior.
static
BoxedString
*
del_str
=
getStaticString
(
"__del__"
);
// TODO: any exceptions here should get caught + printed, instead of causing a std::terminate:
Box
*
func
=
instanceGetattributeSimple
<
NOT_REWRITABLE
>
(
inst
,
del_str
,
NULL
);
if
(
func
)
runtimeCall
(
func
,
ArgPassSpec
(
0
),
NULL
,
NULL
,
NULL
,
NULL
,
NULL
);
}
static
Box
*
_instanceBinary
(
Box
*
_inst
,
Box
*
other
,
BoxedString
*
attr
)
{
RELEASE_ASSERT
(
_inst
->
cls
==
instance_cls
,
""
);
BoxedInstance
*
inst
=
static_cast
<
BoxedInstance
*>
(
_inst
);
...
...
@@ -1640,6 +1634,128 @@ extern "C" int PyMethod_ClearFreeList(void) noexcept {
return
0
;
// number of entries cleared
}
void
BoxedInstance
::
dealloc
(
Box
*
b
)
noexcept
{
RELEASE_ASSERT
(
b
->
cls
==
instance_cls
,
""
);
BoxedInstance
*
inst
=
static_cast
<
BoxedInstance
*>
(
b
);
PyObject
*
error_type
,
*
error_value
,
*
error_traceback
;
PyObject
*
del
;
static
PyObject
*
delstr
;
_PyObject_GC_UNTRACK
(
inst
);
if
(
inst
->
weakreflist
!=
NULL
)
PyObject_ClearWeakRefs
((
PyObject
*
)
inst
);
/* Temporarily resurrect the object. */
assert
(
inst
->
cls
==
&
PyInstance_Type
);
assert
(
inst
->
ob_refcnt
==
0
);
inst
->
ob_refcnt
=
1
;
/* Save the current exception, if any. */
PyErr_Fetch
(
&
error_type
,
&
error_value
,
&
error_traceback
);
/* Execute __del__ method, if any. */
if
(
delstr
==
NULL
)
{
delstr
=
getStaticString
(
"__del__"
);
if
(
delstr
==
NULL
)
PyErr_WriteUnraisable
((
PyObject
*
)
inst
);
}
// if (delstr && (del = instance_getattr2(inst, delstr)) != NULL) {
// TODO: not sure if this is the same as cpython's getattr2 (and the exception style might be different too?)
if
(
delstr
&&
(
del
=
instanceGetattributeSimple
<
NOT_REWRITABLE
>
(
inst
,
static_cast
<
BoxedString
*>
(
delstr
),
NULL
))
!=
NULL
)
{
PyObject
*
res
=
PyEval_CallObject
(
del
,
(
PyObject
*
)
NULL
);
if
(
res
==
NULL
)
PyErr_WriteUnraisable
(
del
);
else
Py_DECREF
(
res
);
Py_DECREF
(
del
);
}
/* Restore the saved exception. */
PyErr_Restore
(
error_type
,
error_value
,
error_traceback
);
/* Undo the temporary resurrection; can't use DECREF here, it would
* cause a recursive call.
*/
assert
(
inst
->
ob_refcnt
>
0
);
if
(
--
inst
->
ob_refcnt
==
0
)
{
/* New weakrefs could be created during the finalizer call.
If this occurs, clear them out without calling their
finalizers since they might rely on part of the object
being finalized that has already been destroyed. */
while
(
inst
->
weakreflist
!=
NULL
)
{
_PyWeakref_ClearRef
((
PyWeakReference
*
)(
inst
->
weakreflist
));
}
Py_DECREF
(
inst
->
inst_cls
);
inst
->
attrs
.
clear
();
PyObject_GC_Del
(
inst
);
}
else
{
Py_ssize_t
refcnt
=
inst
->
ob_refcnt
;
/* __del__ resurrected it! Make it look like the original
* Py_DECREF never happened.
*/
_Py_NewReference
((
PyObject
*
)
inst
);
inst
->
ob_refcnt
=
refcnt
;
_PyObject_GC_TRACK
(
inst
);
/* If Py_REF_DEBUG, _Py_NewReference bumped _Py_RefTotal, so
* we need to undo that. */
_Py_DEC_REFTOTAL
;
/* If Py_TRACE_REFS, _Py_NewReference re-added self to the
* object chain, so no more to do there.
* If COUNT_ALLOCS, the original decref bumped tp_frees, and
* _Py_NewReference bumped tp_allocs: both of those need to be
* undone.
*/
#ifdef COUNT_ALLOCS
--
inst
->
cls
->
tp_frees
;
--
inst
->
cls
->
tp_allocs
;
#endif
}
}
int
BoxedInstance
::
traverse
(
Box
*
o
,
visitproc
visit
,
void
*
arg
)
noexcept
{
BoxedInstance
*
self
=
static_cast
<
BoxedInstance
*>
(
o
);
Py_VISIT_HCATTRS
(
self
->
attrs
);
Py_VISIT
(
self
->
inst_cls
);
return
0
;
}
int
BoxedInstance
::
clear
(
Box
*
self
)
noexcept
{
Py_FatalError
(
"unimplemented"
);
}
void
BoxedClassobj
::
dealloc
(
Box
*
b
)
noexcept
{
_PyObject_GC_UNTRACK
(
b
);
BoxedClassobj
*
cl
=
static_cast
<
BoxedClassobj
*>
(
b
);
Py_DECREF
(
cl
->
bases
);
Py_DECREF
(
cl
->
name
);
if
(
cl
->
weakreflist
)
PyObject_ClearWeakRefs
(
cl
);
cl
->
clearAttrs
();
cl
->
cls
->
tp_free
(
cl
);
}
int
BoxedClassobj
::
traverse
(
Box
*
o
,
visitproc
visit
,
void
*
arg
)
noexcept
{
BoxedClassobj
*
cl
=
static_cast
<
BoxedClassobj
*>
(
o
);
Py_VISIT
(
cl
->
bases
);
Py_VISIT
(
cl
->
name
);
Py_VISIT_HCATTRS
(
cl
->
attrs
);
return
0
;
}
int
BoxedClassobj
::
clear
(
Box
*
self
)
noexcept
{
Py_FatalError
(
"unimplemented"
);
}
void
setupClassobj
()
{
classobj_cls
=
BoxedClass
::
create
(
type_cls
,
object_cls
,
offsetof
(
BoxedClassobj
,
attrs
),
offsetof
(
BoxedClassobj
,
weakreflist
),
...
...
@@ -1791,7 +1907,6 @@ void setupClassobj() {
instance_cls
->
tp_as_number
->
nb_index
=
instance_index
;
instance_cls
->
tp_as_number
->
nb_power
=
instance_pow
;
instance_cls
->
tp_as_number
->
nb_inplace_power
=
instance_ipow
;
instance_cls
->
tp_dealloc
=
instance_dealloc
;
instance_cls
->
has_safe_tp_dealloc
=
false
;
}
}
src/runtime/classobj.h
View file @
0b1acea8
...
...
@@ -48,17 +48,14 @@ public:
Box
**
weakreflist
;
BoxedClassobj
(
BoxedString
*
name
,
BoxedTuple
*
bases
)
:
bases
(
bases
),
name
(
name
)
{}
static
void
dealloc
(
Box
*
b
)
noexcept
{
Py_FatalError
(
"unimplemented"
);
}
static
int
traverse
(
Box
*
self
,
visitproc
visit
,
void
*
arg
)
noexcept
{
Py_FatalError
(
"unimplemented"
);
}
static
int
clear
(
Box
*
self
)
noexcept
{
Py_FatalError
(
"unimplemented"
);
BoxedClassobj
(
BoxedString
*
name
,
BoxedTuple
*
bases
)
:
bases
(
bases
),
name
(
name
)
{
Py_INCREF
(
name
);
Py_INCREF
(
bases
);
}
static
void
dealloc
(
Box
*
b
)
noexcept
;
static
int
traverse
(
Box
*
self
,
visitproc
visit
,
void
*
arg
)
noexcept
;
static
int
clear
(
Box
*
self
)
noexcept
;
};
class
BoxedInstance
:
public
Box
{
...
...
@@ -69,19 +66,15 @@ public:
Box
**
weakreflist
;
BoxedInstance
(
BoxedClassobj
*
inst_cls
)
:
inst_cls
(
inst_cls
)
{}
BoxedInstance
(
BoxedClassobj
*
inst_cls
)
:
inst_cls
(
inst_cls
)
{
Py_INCREF
(
inst_cls
);
}
DEFAULT_CLASS
(
instance_cls
);
static
void
dealloc
(
Box
*
b
)
noexcept
{
Py_FatalError
(
"unimplemented"
);
}
static
int
traverse
(
Box
*
self
,
visitproc
visit
,
void
*
arg
)
noexcept
{
Py_FatalError
(
"unimplemented"
);
}
static
int
clear
(
Box
*
self
)
noexcept
{
Py_FatalError
(
"unimplemented"
);
}
static
void
dealloc
(
Box
*
b
)
noexcept
;
static
int
traverse
(
Box
*
self
,
visitproc
visit
,
void
*
arg
)
noexcept
;
static
int
clear
(
Box
*
self
)
noexcept
;
};
Box
*
instance_getattro
(
Box
*
cls
,
Box
*
attr
)
noexcept
;
...
...
@@ -89,7 +82,7 @@ int instance_setattro(Box* cls, Box* attr, Box* value) noexcept;
class
GetattrRewriteArgs
;
template
<
ExceptionStyle
S
>
Box
*
instanceGetattroInternal
(
Box
*
self
,
Box
*
attr
,
GetattrRewriteArgs
*
rewrite_args
)
noexcept
(
S
==
CAPI
);
Box
*
instanceSetattroInternal
(
Box
*
self
,
Box
*
attr
,
Box
*
val
,
SetattrRewriteArgs
*
rewrite_args
);
void
instanceSetattroInternal
(
Box
*
self
,
STOLEN
(
Box
*
)
attr
,
Box
*
val
,
SetattrRewriteArgs
*
rewrite_args
);
}
#endif
src/runtime/objmodel.cpp
View file @
0b1acea8
...
...
@@ -779,7 +779,7 @@ BoxedClass* BoxedClass::create(BoxedClass* metaclass, BoxedClass* base, int attr
void
BoxedClass
::
finishInitialization
()
{
assert
(
!
this
->
tp_dict
);
this
->
tp_dict
=
incref
(
this
->
getAttrWrapper
()
);
this
->
tp_dict
=
this
->
getAttrWrapper
(
);
commonClassSetup
(
this
);
tp_flags
|=
Py_TPFLAGS_READY
;
...
...
@@ -992,7 +992,7 @@ BoxedDict* Box::getDict() {
}
assert
(
d
->
cls
==
dict_cls
);
return
d
;
return
incref
(
d
)
;
}
static
StatCounter
box_getattr_slowpath
(
"slowpath_box_getattr"
);
...
...
@@ -2838,7 +2838,6 @@ extern "C" void setattr(Box* obj, BoxedString* attr, STOLEN(Box*) attr_val) {
RewriterVar
*
r_setattr
;
if
(
tp_setattro
==
instance_setattro
)
{
assert
(
0
&&
"check refcounting"
);
if
(
rewriter
.
get
())
{
SetattrRewriteArgs
rewrite_args
(
rewriter
.
get
(),
rewriter
->
getArg
(
0
),
rewriter
->
getArg
(
2
));
instanceSetattroInternal
(
obj
,
attr
,
attr_val
,
&
rewrite_args
);
...
...
src/runtime/types.cpp
View file @
0b1acea8
...
...
@@ -1305,7 +1305,7 @@ Box* typeCall(Box* obj, BoxedTuple* vararg, BoxedDict* kwargs) {
static
Box
*
typeDict
(
Box
*
obj
,
void
*
context
)
{
if
(
obj
->
cls
->
instancesHaveHCAttrs
())
return
PyDictProxy_New
(
obj
->
getAttrWrapper
(
));
return
PyDictProxy_New
(
autoDecref
(
obj
->
getAttrWrapper
()
));
abort
();
}
...
...
@@ -1392,6 +1392,7 @@ extern "C" Box* createUserClass(BoxedString* name, Box* _bases, Box* _attr_dict)
metaclass
=
bases
->
elts
[
0
]
->
cls
;
}
else
{
Box
*
gl
=
getGlobalsDict
();
AUTO_DECREF
(
gl
);
metaclass
=
PyDict_GetItemString
(
gl
,
"__metaclass__"
);
if
(
!
metaclass
)
{
...
...
@@ -1411,8 +1412,10 @@ extern "C" Box* createUserClass(BoxedString* name, Box* _bases, Box* _attr_dict)
// But for classes created from Python, we don't have this extra untracked reference.
// Rather than fix up the plumbing for now, just reach into the other system and remove this
// class from the list.
RELEASE_ASSERT
(
classes
.
back
()
==
r
,
""
);
classes
.
pop_back
();
if
(
isSubclass
(
r
->
cls
,
type_cls
))
{
RELEASE_ASSERT
(
classes
.
back
()
==
r
,
""
);
classes
.
pop_back
();
}
return
r
;
}
catch
(
ExcInfo
e
)
{
...
...
@@ -2559,7 +2562,7 @@ Box* Box::getAttrWrapper() {
HiddenClass
*
hcls
=
attrs
->
hcls
;
if
(
hcls
->
type
==
HiddenClass
::
DICT_BACKED
)
{
return
attrs
->
attr_list
->
attrs
[
0
]
;
return
incref
(
attrs
->
attr_list
->
attrs
[
0
])
;
}
int
offset
=
hcls
->
getAttrwrapperOffset
();
...
...
@@ -2569,17 +2572,15 @@ Box* Box::getAttrWrapper() {
auto
new_hcls
=
hcls
->
getAttrwrapperChild
();
appendNewHCAttr
(
aw
,
NULL
);
attrs
->
hcls
=
new_hcls
;
Py_DECREF
(
aw
);
return
aw
;
}
else
{
assert
(
hcls
->
type
==
HiddenClass
::
SINGLETON
);
appendNewHCAttr
(
aw
,
NULL
);
hcls
->
appendAttrwrapper
();
Py_DECREF
(
aw
);
return
aw
;
}
}
return
attrs
->
attr_list
->
attrs
[
offset
]
;
return
incref
(
attrs
->
attr_list
->
attrs
[
offset
])
;
}
extern
"C"
PyObject
*
PyObject_GetAttrWrapper
(
PyObject
*
obj
)
noexcept
{
...
...
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