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
0f1eb426
Commit
0f1eb426
authored
Mar 09, 2016
by
Kevin Modzelewski
1
Browse files
Options
Browse Files
Download
Plain Diff
Merge pull request #1109 from undingen/binop_subclass
implement correct binop lookup for subclasses
parents
4f3885c4
e14f5cdc
Changes
2
Hide whitespace changes
Inline
Side-by-side
Showing
2 changed files
with
75 additions
and
87 deletions
+75
-87
src/runtime/objmodel.cpp
src/runtime/objmodel.cpp
+75
-83
test/tests/binops_subclass.py
test/tests/binops_subclass.py
+0
-4
No files found.
src/runtime/objmodel.cpp
View file @
0f1eb426
...
@@ -4474,121 +4474,112 @@ extern "C" Box* runtimeCallCapi(Box* obj, ArgPassSpec argspec, Box* arg1, Box* a
...
@@ -4474,121 +4474,112 @@ extern "C" Box* runtimeCallCapi(Box* obj, ArgPassSpec argspec, Box* arg1, Box* a
}
}
template
<
Rewritable
rewritable
>
template
<
Rewritable
rewritable
>
Box
*
binopInternal
(
Box
*
lhs
,
Box
*
rhs
,
int
op_type
,
bool
inplace
,
BinopRewriteArgs
*
rewrite_args
)
{
static
Box
*
binopInternalHelper
(
BinopRewriteArgs
*&
rewrite_args
,
BoxedString
*
op_name
,
Box
*
lhs
,
Box
*
rhs
,
RewriterVar
*
r_lhs
,
RewriterVar
*
r_rhs
)
{
if
(
rewritable
==
NOT_REWRITABLE
)
{
if
(
rewritable
==
NOT_REWRITABLE
)
{
assert
(
!
rewrite_args
);
assert
(
!
rewrite_args
);
rewrite_args
=
NULL
;
rewrite_args
=
NULL
;
}
}
// TODO handle the case of the rhs being a subclass of the lhs
// this could get really annoying because you can dynamically make one type a subclass
// of the other!
assert
(
gc
::
isValidGCObject
(
lhs
));
assert
(
gc
::
isValidGCObject
(
rhs
));
if
(
rewrite_args
)
{
// TODO probably don't need to guard on the lhs_cls since it
// will get checked no matter what, but the check that should be
// removed is probably the later one.
// ie we should have some way of specifying what we know about the values
// of objects and their attributes, and the attributes' attributes.
rewrite_args
->
lhs
->
addAttrGuard
(
offsetof
(
Box
,
cls
),
(
intptr_t
)
lhs
->
cls
);
rewrite_args
->
rhs
->
addAttrGuard
(
offsetof
(
Box
,
cls
),
(
intptr_t
)
rhs
->
cls
);
}
struct
NotImplementedHelper
{
struct
NotImplementedHelper
{
static
void
call
(
Box
*
r
,
bool
was_notimplemented
)
{
assert
((
r
==
NotImplemented
)
==
was_notimplemented
);
}
static
void
call
(
Box
*
r
,
bool
was_notimplemented
)
{
assert
((
r
==
NotImplemented
)
==
was_notimplemented
);
}
};
};
Box
*
irtn
=
NULL
;
Box
*
rtn
=
NULL
;
if
(
inplace
)
{
BoxedString
*
iop_name
=
getInplaceOpName
(
op_type
);
if
(
rewrite_args
)
{
CallattrRewriteArgs
srewrite_args
(
rewrite_args
->
rewriter
,
rewrite_args
->
lhs
,
rewrite_args
->
destination
);
srewrite_args
.
arg1
=
rewrite_args
->
rhs
;
srewrite_args
.
args_guarded
=
true
;
irtn
=
callattrInternal1
<
CXX
,
REWRITABLE
>
(
lhs
,
iop_name
,
CLASS_ONLY
,
&
srewrite_args
,
ArgPassSpec
(
1
),
rhs
);
if
(
!
srewrite_args
.
isSuccessful
())
{
rewrite_args
=
NULL
;
}
else
if
(
irtn
)
{
rewrite_args
->
out_rtn
=
srewrite_args
.
getReturn
(
ReturnConvention
::
HAS_RETURN
);
// If we allowed a rewrite to get here, it means that we assumed that the class will return NotImplemented
// or not based only on the types of the inputs.
#ifndef NDEBUG
rewrite_args
->
rewriter
->
call
(
false
,
(
void
*
)
NotImplementedHelper
::
call
,
rewrite_args
->
out_rtn
,
rewrite_args
->
rewriter
->
loadConst
(
irtn
==
NotImplemented
));
#endif
}
else
{
srewrite_args
.
assertReturnConvention
(
ReturnConvention
::
NO_RETURN
);
}
}
else
{
irtn
=
callattrInternal1
<
CXX
,
NOT_REWRITABLE
>
(
lhs
,
iop_name
,
CLASS_ONLY
,
NULL
,
ArgPassSpec
(
1
),
rhs
);
}
if
(
irtn
)
{
if
(
irtn
!=
NotImplemented
)
{
if
(
rewrite_args
)
{
assert
(
rewrite_args
->
out_rtn
);
rewrite_args
->
out_success
=
true
;
}
return
irtn
;
}
else
{
assert
(
!
rewrite_args
);
}
}
}
BoxedString
*
op_name
=
getOpName
(
op_type
);
Box
*
lrtn
;
if
(
rewrite_args
)
{
if
(
rewrite_args
)
{
CallattrRewriteArgs
srewrite_args
(
rewrite_args
->
rewriter
,
rewrite_args
->
lhs
,
rewrite_args
->
destination
);
CallattrRewriteArgs
srewrite_args
(
rewrite_args
->
rewriter
,
r_lhs
,
rewrite_args
->
destination
);
srewrite_args
.
arg1
=
rewrite_args
->
rhs
;
srewrite_args
.
arg1
=
r_rhs
;
lrtn
=
callattrInternal1
<
CXX
,
REWRITABLE
>
(
lhs
,
op_name
,
CLASS_ONLY
,
&
srewrite_args
,
ArgPassSpec
(
1
),
rhs
);
srewrite_args
.
args_guarded
=
true
;
rtn
=
callattrInternal1
<
CXX
,
REWRITABLE
>
(
lhs
,
op_name
,
CLASS_ONLY
,
&
srewrite_args
,
ArgPassSpec
(
1
),
rhs
);
if
(
!
srewrite_args
.
isSuccessful
())
{
if
(
!
srewrite_args
.
isSuccessful
())
{
rewrite_args
=
NULL
;
rewrite_args
=
NULL
;
}
else
if
(
l
rtn
)
{
}
else
if
(
rtn
)
{
rewrite_args
->
out_rtn
=
srewrite_args
.
getReturn
(
ReturnConvention
::
HAS_RETURN
);
rewrite_args
->
out_rtn
=
srewrite_args
.
getReturn
(
ReturnConvention
::
HAS_RETURN
);
// If we allowed a rewrite to get here, it means that we assumed that the class will return NotImplemented
// If we allowed a rewrite to get here, it means that we assumed that the class will return NotImplemented
// or not based only on the types of the inputs.
// or not based only on the types of the inputs.
#ifndef NDEBUG
#ifndef NDEBUG
rewrite_args
->
rewriter
->
call
(
false
,
(
void
*
)
NotImplementedHelper
::
call
,
rewrite_args
->
out_rtn
,
rewrite_args
->
rewriter
->
call
(
false
,
(
void
*
)
NotImplementedHelper
::
call
,
rewrite_args
->
out_rtn
,
rewrite_args
->
rewriter
->
loadConst
(
i
rtn
==
NotImplemented
));
rewrite_args
->
rewriter
->
loadConst
(
rtn
==
NotImplemented
));
#endif
#endif
}
else
{
}
else
{
srewrite_args
.
assertReturnConvention
(
ReturnConvention
::
NO_RETURN
);
srewrite_args
.
assertReturnConvention
(
ReturnConvention
::
NO_RETURN
);
}
}
}
else
{
lrtn
=
callattrInternal1
<
CXX
,
NOT_REWRITABLE
>
(
lhs
,
op_name
,
CLASS_ONLY
,
NULL
,
ArgPassSpec
(
1
),
rhs
);
}
if
(
rewrite_args
&&
rtn
)
{
if
(
lrtn
)
{
if
(
rtn
!=
NotImplemented
)
if
(
lrtn
!=
NotImplemented
)
{
if
(
rewrite_args
)
{
assert
(
rewrite_args
->
out_rtn
);
rewrite_args
->
out_success
=
true
;
rewrite_args
->
out_success
=
true
;
else
{
// I think our guarding up to here is correct; the issue is we won't be able to complete
// the rewrite since we have more guards to do, but we already did some mutations.
rewrite_args
->
out_success
=
false
;
rewrite_args
=
NULL
;
REWRITE_ABORTED
(
""
);
}
}
return
lrtn
;
}
}
// we don't need to abort the rewrite when the attribute does not exist (rtn==null) because we only rewrite
// binops when both sides are not user defined types for which we assume that they will never change.
}
else
{
rtn
=
callattrInternal1
<
CXX
,
NOT_REWRITABLE
>
(
lhs
,
op_name
,
CLASS_ONLY
,
NULL
,
ArgPassSpec
(
1
),
rhs
);
}
}
// TODO patch these cases
return
rtn
;
// actually, I think our guarding up to here is correct; the issue is we won't be able to complete
}
// the rewrite since we have more guards to do, but we already did some mutations.
if
(
rewrite_args
)
{
template
<
Rewritable
rewritable
>
assert
(
rewrite_args
->
out_success
==
false
);
Box
*
binopInternal
(
Box
*
lhs
,
Box
*
rhs
,
int
op_type
,
bool
inplace
,
BinopRewriteArgs
*
rewrite_args
)
{
if
(
rewritable
==
NOT_REWRITABLE
)
{
assert
(
!
rewrite_args
);
rewrite_args
=
NULL
;
rewrite_args
=
NULL
;
REWRITE_ABORTED
(
""
);
}
}
BoxedString
*
rop_name
=
getReverseOpName
(
op_type
);
assert
(
gc
::
isValidGCObject
(
lhs
));
Box
*
rrtn
=
callattrInternal1
<
CXX
,
REWRITABLE
>
(
rhs
,
rop_name
,
CLASS_ONLY
,
NULL
,
ArgPassSpec
(
1
),
lhs
);
assert
(
gc
::
isValidGCObject
(
rhs
));
if
(
rrtn
!=
NULL
&&
rrtn
!=
NotImplemented
)
return
rrtn
;
RewriterVar
*
r_lhs
=
NULL
;
RewriterVar
*
r_rhs
=
NULL
;
if
(
rewrite_args
)
{
r_lhs
=
rewrite_args
->
lhs
;
r_rhs
=
rewrite_args
->
rhs
;
RewriterVar
*
r_lhs_cls
=
r_lhs
->
getAttr
(
offsetof
(
Box
,
cls
));
r_lhs_cls
->
addGuard
((
intptr_t
)
lhs
->
cls
);
RewriterVar
*
r_rhs_cls
=
r_rhs
->
getAttr
(
offsetof
(
Box
,
cls
));
r_rhs_cls
->
addGuard
((
intptr_t
)
rhs
->
cls
);
r_lhs_cls
->
addAttrGuard
(
offsetof
(
BoxedClass
,
tp_mro
),
(
intptr_t
)
lhs
->
cls
->
tp_mro
);
r_rhs_cls
->
addAttrGuard
(
offsetof
(
BoxedClass
,
tp_mro
),
(
intptr_t
)
rhs
->
cls
->
tp_mro
);
}
Box
*
irtn
=
NULL
;
if
(
inplace
)
{
BoxedString
*
iop_name
=
getInplaceOpName
(
op_type
);
irtn
=
binopInternalHelper
<
rewritable
>
(
rewrite_args
,
iop_name
,
lhs
,
rhs
,
r_lhs
,
r_rhs
);
if
(
irtn
&&
irtn
!=
NotImplemented
)
return
irtn
;
}
bool
should_try_reverse
=
true
;
Box
*
rrtn
=
NULL
;
if
(
lhs
->
cls
!=
rhs
->
cls
&&
isSubclass
(
rhs
->
cls
,
lhs
->
cls
))
{
should_try_reverse
=
false
;
BoxedString
*
rop_name
=
getReverseOpName
(
op_type
);
rrtn
=
binopInternalHelper
<
rewritable
>
(
rewrite_args
,
rop_name
,
rhs
,
lhs
,
r_rhs
,
r_lhs
);
if
(
rrtn
&&
rrtn
!=
NotImplemented
)
return
rrtn
;
}
BoxedString
*
op_name
=
getOpName
(
op_type
);
Box
*
lrtn
=
binopInternalHelper
<
rewritable
>
(
rewrite_args
,
op_name
,
lhs
,
rhs
,
r_lhs
,
r_rhs
);
if
(
lrtn
&&
lrtn
!=
NotImplemented
)
return
lrtn
;
if
(
should_try_reverse
)
{
BoxedString
*
rop_name
=
getReverseOpName
(
op_type
);
rrtn
=
binopInternalHelper
<
rewritable
>
(
rewrite_args
,
rop_name
,
rhs
,
lhs
,
r_rhs
,
r_lhs
);
if
(
rrtn
&&
rrtn
!=
NotImplemented
)
return
rrtn
;
}
llvm
::
StringRef
op_sym
=
getOpSymbol
(
op_type
);
llvm
::
StringRef
op_sym
=
getOpSymbol
(
op_type
);
const
char
*
op_sym_suffix
=
""
;
const
char
*
op_sym_suffix
=
""
;
...
@@ -4597,6 +4588,7 @@ Box* binopInternal(Box* lhs, Box* rhs, int op_type, bool inplace, BinopRewriteAr
...
@@ -4597,6 +4588,7 @@ Box* binopInternal(Box* lhs, Box* rhs, int op_type, bool inplace, BinopRewriteAr
}
}
if
(
VERBOSITY
())
{
if
(
VERBOSITY
())
{
BoxedString
*
rop_name
=
getReverseOpName
(
op_type
);
if
(
inplace
)
{
if
(
inplace
)
{
BoxedString
*
iop_name
=
getInplaceOpName
(
op_type
);
BoxedString
*
iop_name
=
getInplaceOpName
(
op_type
);
if
(
irtn
)
if
(
irtn
)
...
...
test/tests/binops_subclass.py
View file @
0f1eb426
# expected: fail
# - binop ordering is wrong,
# should prefer executing a operation on the subclass first
class
M
(
type
):
class
M
(
type
):
def
__instancecheck__
(
self
,
rhs
):
def
__instancecheck__
(
self
,
rhs
):
print
"M.instancecheck"
,
print
"M.instancecheck"
,
...
...
Boxiang Sun
@Daetalus
mentioned in commit
5d803aff
·
Sep 08, 2016
mentioned in commit
5d803aff
mentioned in commit 5d803affc9792921e4146bd45b8c182db1c95e50
Toggle commit list
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