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
5a0f66dd
Commit
5a0f66dd
authored
Jul 20, 2015
by
Kevin Modzelewski
Browse files
Options
Browse Files
Download
Plain Diff
Merge pull request #729 from kmod/attr_perf
more small perf work
parents
4ba2dfed
c78fdcb9
Changes
11
Show whitespace changes
Inline
Side-by-side
Showing
11 changed files
with
94 additions
and
69 deletions
+94
-69
Makefile
Makefile
+2
-2
microbenchmarks/django_variable.py
microbenchmarks/django_variable.py
+8
-0
microbenchmarks/getattrfunc_ubench.py
microbenchmarks/getattrfunc_ubench.py
+10
-1
src/asm_writing/icinfo.cpp
src/asm_writing/icinfo.cpp
+4
-6
src/asm_writing/icinfo.h
src/asm_writing/icinfo.h
+5
-3
src/asm_writing/rewriter.cpp
src/asm_writing/rewriter.cpp
+27
-21
src/asm_writing/rewriter.h
src/asm_writing/rewriter.h
+20
-14
src/core/stringpool.cpp
src/core/stringpool.cpp
+2
-13
src/core/stringpool.h
src/core/stringpool.h
+0
-6
src/runtime/builtin_modules/pyston.cpp
src/runtime/builtin_modules/pyston.cpp
+2
-0
src/runtime/objmodel.cpp
src/runtime/objmodel.cpp
+14
-3
No files found.
Makefile
View file @
5a0f66dd
...
@@ -1033,9 +1033,9 @@ $(call make_target,_gcc)
...
@@ -1033,9 +1033,9 @@ $(call make_target,_gcc)
$(call
make_target,_release_gcc)
$(call
make_target,_release_gcc)
nosearch_runpy_% nosearch_pyrun_%
:
%.py ext_python
nosearch_runpy_% nosearch_pyrun_%
:
%.py ext_python
$(VERB)
PYTHONPATH
=
test
/test_extension/build/lib.linux-x86_64-2.7 zsh
-c
'time
$(CPYTHON)
$<'
$(VERB)
PYTHONPATH
=
test
/test_extension/build/lib.linux-x86_64-2.7
:
$
${PYTHONPATH}
zsh
-c
'time
$(CPYTHON)
$<'
nosearch_pypyrun_%
:
%.py ext_python
nosearch_pypyrun_%
:
%.py ext_python
$(VERB)
PYTHONPATH
=
test
/test_extension/build/lib.linux-x86_64-2.7 zsh
-c
'time
$(PYPY)
$<'
$(VERB)
PYTHONPATH
=
test
/test_extension/build/lib.linux-x86_64-2.7
:
$
${PYTHONPATH}
zsh
-c
'time
$(PYPY)
$<'
$(call
make_search,runpy_%)
$(call
make_search,runpy_%)
$(call
make_search,pyrun_%)
$(call
make_search,pyrun_%)
$(call
make_search,pypyrun_%)
$(call
make_search,pypyrun_%)
...
...
microbenchmarks/django_variable.py
0 → 100644
View file @
5a0f66dd
import
os
import
sys
sys
.
path
.
append
(
os
.
path
.
join
(
os
.
path
.
dirname
(
__file__
),
"../integration/django"
))
from
django.template.base
import
Variable
for
i
in
xrange
(
400000
):
Variable
(
u"model.name"
)
microbenchmarks/getattrfunc_ubench.py
View file @
5a0f66dd
...
@@ -5,6 +5,15 @@ def f():
...
@@ -5,6 +5,15 @@ def f():
g
=
getattr
g
=
getattr
c
=
C
()
c
=
C
()
c
.
o
=
1
c
.
o
=
1
for
i
in
xrange
(
10000000
):
for
i
in
xrange
(
1000000
):
g
(
c
,
"o"
)
g
(
c
,
"o"
)
g
(
c
,
"o"
)
g
(
c
,
"o"
)
g
(
c
,
"o"
)
g
(
c
,
"o"
)
g
(
c
,
"o"
)
g
(
c
,
"o"
)
g
(
c
,
"o"
)
g
(
c
,
"o"
)
g
(
c
,
"o"
)
f
()
f
()
src/asm_writing/icinfo.cpp
View file @
5a0f66dd
...
@@ -56,17 +56,15 @@ void ICSlotInfo::clear() {
...
@@ -56,17 +56,15 @@ void ICSlotInfo::clear() {
ic
->
clear
(
this
);
ic
->
clear
(
this
);
}
}
ICSlotRewrite
::
ICSlotRewrite
(
ICInfo
*
ic
,
const
char
*
debug_name
)
:
ic
(
ic
),
debug_name
(
debug_name
)
{
ICSlotRewrite
::
ICSlotRewrite
(
ICInfo
*
ic
,
const
char
*
debug_name
)
buf
=
(
uint8_t
*
)
malloc
(
ic
->
getSlotSize
());
:
ic
(
ic
),
debug_name
(
debug_name
),
buf
((
uint8_t
*
)
malloc
(
ic
->
getSlotSize
())),
assembler
(
buf
,
ic
->
getSlotSize
())
{
assembler
=
new
Assembler
(
buf
,
ic
->
getSlotSize
());
assembler
.
nop
();
assembler
->
nop
();
if
(
VERBOSITY
()
>=
4
)
if
(
VERBOSITY
()
>=
4
)
printf
(
"starting %s icentry
\n
"
,
debug_name
);
printf
(
"starting %s icentry
\n
"
,
debug_name
);
}
}
ICSlotRewrite
::~
ICSlotRewrite
()
{
ICSlotRewrite
::~
ICSlotRewrite
()
{
delete
assembler
;
free
(
buf
);
free
(
buf
);
}
}
...
@@ -109,7 +107,7 @@ void ICSlotRewrite::commit(CommitHook* hook) {
...
@@ -109,7 +107,7 @@ void ICSlotRewrite::commit(CommitHook* hook) {
if
(
!
do_commit
)
if
(
!
do_commit
)
return
;
return
;
assert
(
!
assembler
->
hasFailed
());
assert
(
!
assembler
.
hasFailed
());
for
(
int
i
=
0
;
i
<
dependencies
.
size
();
i
++
)
{
for
(
int
i
=
0
;
i
<
dependencies
.
size
();
i
++
)
{
ICInvalidator
*
invalidator
=
dependencies
[
i
].
first
;
ICInvalidator
*
invalidator
=
dependencies
[
i
].
first
;
...
...
src/asm_writing/icinfo.h
View file @
5a0f66dd
...
@@ -21,6 +21,7 @@
...
@@ -21,6 +21,7 @@
#include "llvm/IR/CallingConv.h"
#include "llvm/IR/CallingConv.h"
#include "asm_writing/assembler.h"
#include "asm_writing/types.h"
#include "asm_writing/types.h"
namespace
pyston
{
namespace
pyston
{
...
@@ -53,12 +54,13 @@ public:
...
@@ -53,12 +54,13 @@ public:
private:
private:
ICInfo
*
ic
;
ICInfo
*
ic
;
assembler
::
Assembler
*
assembler
;
const
char
*
debug_name
;
const
char
*
debug_name
;
uint8_t
*
buf
;
uint8_t
*
buf
;
std
::
vector
<
std
::
pair
<
ICInvalidator
*
,
int64_t
>>
dependencies
;
assembler
::
Assembler
assembler
;
llvm
::
SmallVector
<
std
::
pair
<
ICInvalidator
*
,
int64_t
>
,
4
>
dependencies
;
ICSlotInfo
*
ic_entry
;
ICSlotInfo
*
ic_entry
;
...
@@ -66,7 +68,7 @@ public:
...
@@ -66,7 +68,7 @@ public:
ICSlotRewrite
(
ICInfo
*
ic
,
const
char
*
debug_name
);
ICSlotRewrite
(
ICInfo
*
ic
,
const
char
*
debug_name
);
~
ICSlotRewrite
();
~
ICSlotRewrite
();
assembler
::
Assembler
*
getAssembler
()
{
return
assembler
;
}
assembler
::
Assembler
*
getAssembler
()
{
return
&
assembler
;
}
int
getSlotSize
();
int
getSlotSize
();
int
getScratchRspOffset
();
int
getScratchRspOffset
();
int
getScratchSize
();
int
getScratchSize
();
...
...
src/asm_writing/rewriter.cpp
View file @
5a0f66dd
...
@@ -185,9 +185,11 @@ void Rewriter::ConstLoader::moveImmediate(uint64_t val, assembler::Register dst_
...
@@ -185,9 +185,11 @@ void Rewriter::ConstLoader::moveImmediate(uint64_t val, assembler::Register dst_
assembler
::
Register
Rewriter
::
ConstLoader
::
findConst
(
uint64_t
val
,
bool
&
found_value
)
{
assembler
::
Register
Rewriter
::
ConstLoader
::
findConst
(
uint64_t
val
,
bool
&
found_value
)
{
assert
(
rewriter
->
phase_emitting
);
assert
(
rewriter
->
phase_emitting
);
auto
it
=
constToVar
.
find
(
val
);
for
(
auto
&
p
:
consts
)
{
if
(
it
!=
constToVar
.
end
())
{
if
(
p
.
first
!=
val
)
RewriterVar
*
var
=
it
->
second
;
continue
;
RewriterVar
*
var
=
p
.
second
;
for
(
Location
l
:
var
->
locations
)
{
for
(
Location
l
:
var
->
locations
)
{
if
(
l
.
type
==
Location
::
Register
)
{
if
(
l
.
type
==
Location
::
Register
)
{
found_value
=
true
;
found_value
=
true
;
...
@@ -216,9 +218,9 @@ assembler::Register Rewriter::ConstLoader::loadConst(uint64_t val, Location othe
...
@@ -216,9 +218,9 @@ assembler::Register Rewriter::ConstLoader::loadConst(uint64_t val, Location othe
assert
(
rewriter
->
phase_emitting
);
assert
(
rewriter
->
phase_emitting
);
bool
found_value
=
false
;
bool
found_value
=
false
;
assembler
::
Register
reg
=
findConst
(
val
,
found_value
);
assembler
::
Register
/*
reg = findConst(val, found_value);
if (found_value)
if (found_value)
return
reg
;
return reg;
*/
reg
=
rewriter
->
allocReg
(
Location
::
any
(),
otherThan
);
reg
=
rewriter
->
allocReg
(
Location
::
any
(),
otherThan
);
if
(
tryLea
(
val
,
reg
))
if
(
tryLea
(
val
,
reg
))
...
@@ -711,10 +713,15 @@ void Rewriter::_trap() {
...
@@ -711,10 +713,15 @@ void Rewriter::_trap() {
RewriterVar
*
Rewriter
::
loadConst
(
int64_t
val
,
Location
dest
)
{
RewriterVar
*
Rewriter
::
loadConst
(
int64_t
val
,
Location
dest
)
{
STAT_TIMER
(
t0
,
"us_timer_rewriter"
,
10
);
STAT_TIMER
(
t0
,
"us_timer_rewriter"
,
10
);
RewriterVar
*&
const_loader_var
=
const_loader
.
constToVar
[
val
];
for
(
auto
&
p
:
const_loader
.
consts
)
{
if
(
!
const_loader_var
)
{
if
(
p
.
first
!=
val
)
const_loader_var
=
createNewConstantVar
(
val
);
continue
;
return
p
.
second
;
}
}
RewriterVar
*
const_loader_var
=
createNewConstantVar
(
val
);
const_loader
.
consts
.
push_back
(
std
::
make_pair
(
val
,
const_loader_var
));
return
const_loader_var
;
return
const_loader_var
;
}
}
...
@@ -1099,11 +1106,11 @@ void Rewriter::commit() {
...
@@ -1099,11 +1106,11 @@ void Rewriter::commit() {
for
(
int
i
=
0
;
i
<
live_outs
.
size
();
i
++
)
{
for
(
int
i
=
0
;
i
<
live_outs
.
size
();
i
++
)
{
live_outs
[
i
]
->
uses
.
push_back
(
actions
.
size
());
live_outs
[
i
]
->
uses
.
push_back
(
actions
.
size
());
}
}
for
(
RewriterVar
*
var
:
vars
)
{
for
(
RewriterVar
&
var
:
vars
)
{
// Add a use for every constant. This helps make constants available for the lea stuff
// Add a use for every constant. This helps make constants available for the lea stuff
// But since "spilling" a constant has no cost, it shouldn't add register pressure.
// But since "spilling" a constant has no cost, it shouldn't add register pressure.
if
(
var
->
is_constant
)
{
if
(
var
.
is_constant
)
{
var
->
uses
.
push_back
(
actions
.
size
());
var
.
uses
.
push_back
(
actions
.
size
());
}
}
}
}
...
@@ -1176,22 +1183,22 @@ void Rewriter::commit() {
...
@@ -1176,22 +1183,22 @@ void Rewriter::commit() {
// Make sure that we have been calling bumpUse correctly.
// Make sure that we have been calling bumpUse correctly.
// All uses should have been accounted for, other than the live outs
// All uses should have been accounted for, other than the live outs
#ifndef NDEBUG
#ifndef NDEBUG
for
(
RewriterVar
*
var
:
vars
)
{
for
(
RewriterVar
&
var
:
vars
)
{
int
num_as_live_out
=
0
;
int
num_as_live_out
=
0
;
for
(
RewriterVar
*
live_out
:
live_outs
)
{
for
(
RewriterVar
*
live_out
:
live_outs
)
{
if
(
live_out
==
var
)
{
if
(
live_out
==
&
var
)
{
num_as_live_out
++
;
num_as_live_out
++
;
}
}
}
}
assert
(
var
->
next_use
+
num_as_live_out
+
(
var
->
is_constant
?
1
:
0
)
==
var
->
uses
.
size
());
assert
(
var
.
next_use
+
num_as_live_out
+
(
var
.
is_constant
?
1
:
0
)
==
var
.
uses
.
size
());
}
}
#endif
#endif
assert
(
live_out_regs
.
size
()
==
live_outs
.
size
());
assert
(
live_out_regs
.
size
()
==
live_outs
.
size
());
for
(
RewriterVar
*
var
:
vars
)
{
for
(
RewriterVar
&
var
:
vars
)
{
if
(
var
->
is_constant
)
{
if
(
var
.
is_constant
)
{
var
->
bumpUse
();
var
.
bumpUse
();
}
}
}
}
...
@@ -1686,9 +1693,8 @@ void Rewriter::removeLocationFromVar(RewriterVar* var, Location l) {
...
@@ -1686,9 +1693,8 @@ void Rewriter::removeLocationFromVar(RewriterVar* var, Location l) {
RewriterVar
*
Rewriter
::
createNewVar
()
{
RewriterVar
*
Rewriter
::
createNewVar
()
{
assertPhaseCollecting
();
assertPhaseCollecting
();
RewriterVar
*
var
=
new
RewriterVar
(
this
);
vars
.
emplace_back
(
this
);
vars
.
push_back
(
var
);
return
&
vars
.
back
();
return
var
;
}
}
RewriterVar
*
Rewriter
::
createNewConstantVar
(
uint64_t
val
)
{
RewriterVar
*
Rewriter
::
createNewConstantVar
(
uint64_t
val
)
{
...
...
src/asm_writing/rewriter.h
View file @
5a0f66dd
...
@@ -15,6 +15,7 @@
...
@@ -15,6 +15,7 @@
#ifndef PYSTON_ASMWRITING_REWRITER_H
#ifndef PYSTON_ASMWRITING_REWRITER_H
#define PYSTON_ASMWRITING_REWRITER_H
#define PYSTON_ASMWRITING_REWRITER_H
#include <deque>
#include <map>
#include <map>
#include <memory>
#include <memory>
#include <tuple>
#include <tuple>
...
@@ -101,6 +102,8 @@ public:
...
@@ -101,6 +102,8 @@ public:
bool
operator
!=
(
const
Location
rhs
)
const
{
return
!
(
*
this
==
rhs
);
}
bool
operator
!=
(
const
Location
rhs
)
const
{
return
!
(
*
this
==
rhs
);
}
bool
operator
<
(
const
Location
&
rhs
)
const
{
return
this
->
asInt
()
<
rhs
.
asInt
();
}
uint64_t
asInt
()
const
{
return
(
int
)
type
+
((
uint64_t
)
_data
<<
4
);
}
uint64_t
asInt
()
const
{
return
(
int
)
type
+
((
uint64_t
)
_data
<<
4
);
}
void
dump
()
const
;
void
dump
()
const
;
...
@@ -224,7 +227,7 @@ public:
...
@@ -224,7 +227,7 @@ public:
private:
private:
Rewriter
*
rewriter
;
Rewriter
*
rewriter
;
std
::
unordered_
set
<
Location
>
locations
;
std
::
set
<
Location
>
locations
;
bool
isInLocation
(
Location
l
);
bool
isInLocation
(
Location
l
);
// uses is a vector of the indices into the Rewriter::actions vector
// uses is a vector of the indices into the Rewriter::actions vector
...
@@ -317,7 +320,7 @@ protected:
...
@@ -317,7 +320,7 @@ protected:
// Loads the constant into any register or if already in a register just return it
// Loads the constant into any register or if already in a register just return it
assembler
::
Register
loadConst
(
uint64_t
val
,
Location
otherThan
=
Location
::
any
());
assembler
::
Register
loadConst
(
uint64_t
val
,
Location
otherThan
=
Location
::
any
());
std
::
unordered_map
<
uint64_t
,
RewriterVar
*>
constToVar
;
std
::
vector
<
std
::
pair
<
uint64_t
,
RewriterVar
*>>
consts
;
};
};
...
@@ -326,7 +329,7 @@ protected:
...
@@ -326,7 +329,7 @@ protected:
ICSlotInfo
*
picked_slot
;
ICSlotInfo
*
picked_slot
;
ConstLoader
const_loader
;
ConstLoader
const_loader
;
std
::
vector
<
RewriterVar
*
>
vars
;
std
::
deque
<
RewriterVar
>
vars
;
const
Location
return_location
;
const
Location
return_location
;
...
@@ -346,11 +349,11 @@ protected:
...
@@ -346,11 +349,11 @@ protected:
void
assertPhaseEmitting
()
{}
void
assertPhaseEmitting
()
{}
#endif
#endif
std
::
vector
<
int
>
live_out_regs
;
llvm
::
SmallVector
<
int
,
8
>
live_out_regs
;
LocMap
<
RewriterVar
*>
vars_by_location
;
LocMap
<
RewriterVar
*>
vars_by_location
;
std
::
vector
<
RewriterVar
*
>
args
;
llvm
::
SmallVector
<
RewriterVar
*
,
8
>
args
;
std
::
vector
<
RewriterVar
*
>
live_outs
;
llvm
::
SmallVector
<
RewriterVar
*
,
8
>
live_outs
;
Rewriter
(
std
::
unique_ptr
<
ICSlotRewrite
>
rewrite
,
int
num_args
,
const
std
::
vector
<
int
>&
live_outs
);
Rewriter
(
std
::
unique_ptr
<
ICSlotRewrite
>
rewrite
,
int
num_args
,
const
std
::
vector
<
int
>&
live_outs
);
...
@@ -443,15 +446,22 @@ protected:
...
@@ -443,15 +446,22 @@ protected:
void
assertConsistent
()
{
void
assertConsistent
()
{
#ifndef NDEBUG
#ifndef NDEBUG
for
(
RewriterVar
*
var
:
vars
)
{
for
(
RewriterVar
&
var
:
vars
)
{
for
(
Location
l
:
var
->
locations
)
{
for
(
Location
l
:
var
.
locations
)
{
assert
(
vars_by_location
[
l
]
==
var
);
assert
(
vars_by_location
[
l
]
==
&
var
);
}
}
}
}
for
(
std
::
pair
<
Location
,
RewriterVar
*>
p
:
vars_by_location
.
getAsMap
())
{
for
(
std
::
pair
<
Location
,
RewriterVar
*>
p
:
vars_by_location
.
getAsMap
())
{
assert
(
p
.
second
!=
NULL
);
assert
(
p
.
second
!=
NULL
);
if
(
p
.
second
!=
LOCATION_PLACEHOLDER
)
{
if
(
p
.
second
!=
LOCATION_PLACEHOLDER
)
{
assert
(
std
::
find
(
vars
.
begin
(),
vars
.
end
(),
p
.
second
)
!=
vars
.
end
());
bool
found
=
false
;
for
(
auto
&
v
:
vars
)
{
if
(
&
v
==
p
.
second
)
{
found
=
true
;
break
;
}
}
assert
(
found
);
assert
(
p
.
second
->
locations
.
count
(
p
.
first
)
==
1
);
assert
(
p
.
second
->
locations
.
count
(
p
.
first
)
==
1
);
}
}
}
}
...
@@ -471,10 +481,6 @@ public:
...
@@ -471,10 +481,6 @@ public:
if
(
!
finished
)
if
(
!
finished
)
this
->
abort
();
this
->
abort
();
assert
(
finished
);
assert
(
finished
);
for
(
RewriterVar
*
var
:
vars
)
{
delete
var
;
}
}
}
Location
getReturnDestination
();
Location
getReturnDestination
();
...
...
src/core/stringpool.cpp
View file @
5a0f66dd
...
@@ -19,19 +19,8 @@
...
@@ -19,19 +19,8 @@
namespace
pyston
{
namespace
pyston
{
InternedString
InternedStringPool
::
get
(
llvm
::
StringRef
arg
)
{
InternedString
InternedStringPool
::
get
(
llvm
::
StringRef
arg
)
{
auto
it
=
interned
.
find
(
arg
);
BoxedString
*
s
;
if
(
it
!=
interned
.
end
())
{
s
=
it
->
second
;
}
else
{
// HACK: should properly track this liveness:
// HACK: should properly track this liveness:
s
=
internStringImmortal
(
arg
);
BoxedString
*
s
=
internStringImmortal
(
arg
);
// Note: make sure the key points to the value we just created, not the
// argument string:
interned
[
s
->
s
()]
=
s
;
}
#ifndef NDEBUG
#ifndef NDEBUG
return
InternedString
(
s
,
this
);
return
InternedString
(
s
,
this
);
...
...
src/core/stringpool.h
View file @
5a0f66dd
...
@@ -89,12 +89,6 @@ public:
...
@@ -89,12 +89,6 @@ public:
};
};
class
InternedStringPool
{
class
InternedStringPool
{
private:
// We probably don't need to pull in llvm::StringRef as the key, but it's better than std::string
// which I assume forces extra allocations.
// (We could define a custom string-pointer container but is it worth it?)
std
::
unordered_map
<
llvm
::
StringRef
,
BoxedString
*>
interned
;
public:
public:
void
gcHandler
(
gc
::
GCVisitor
*
v
);
void
gcHandler
(
gc
::
GCVisitor
*
v
);
InternedString
get
(
llvm
::
StringRef
s
);
InternedString
get
(
llvm
::
StringRef
s
);
...
...
src/runtime/builtin_modules/pyston.cpp
View file @
5a0f66dd
...
@@ -43,6 +43,8 @@ static Box* setOption(Box* option, Box* value) {
...
@@ -43,6 +43,8 @@ static Box* setOption(Box* option, Box* value) {
else
CHECK
(
REOPT_THRESHOLD_BASELINE
);
else
CHECK
(
REOPT_THRESHOLD_BASELINE
);
else
CHECK
(
OSR_THRESHOLD_BASELINE
);
else
CHECK
(
OSR_THRESHOLD_BASELINE
);
else
CHECK
(
SPECULATION_THRESHOLD
);
else
CHECK
(
SPECULATION_THRESHOLD
);
else
CHECK
(
ENABLE_ICS
);
else
CHECK
(
ENABLE_ICGETATTRS
);
else
raiseExcHelper
(
ValueError
,
"unknown option name '%s"
,
option_string
->
data
());
else
raiseExcHelper
(
ValueError
,
"unknown option name '%s"
,
option_string
->
data
());
return
None
;
return
None
;
...
...
src/runtime/objmodel.cpp
View file @
5a0f66dd
...
@@ -714,7 +714,7 @@ Box* Box::getattr(BoxedString* attr, GetattrRewriteArgs* rewrite_args) {
...
@@ -714,7 +714,7 @@ Box* Box::getattr(BoxedString* attr, GetattrRewriteArgs* rewrite_args) {
HCAttrs
*
attrs
=
getHCAttrsPtr
();
HCAttrs
*
attrs
=
getHCAttrsPtr
();
HiddenClass
*
hcls
=
attrs
->
hcls
;
HiddenClass
*
hcls
=
attrs
->
hcls
;
if
(
hcls
->
type
==
HiddenClass
::
DICT_BACKED
)
{
if
(
unlikely
(
hcls
->
type
==
HiddenClass
::
DICT_BACKED
)
)
{
if
(
rewrite_args
)
if
(
rewrite_args
)
assert
(
!
rewrite_args
->
out_success
);
assert
(
!
rewrite_args
->
out_success
);
rewrite_args
=
NULL
;
rewrite_args
=
NULL
;
...
@@ -728,7 +728,7 @@ Box* Box::getattr(BoxedString* attr, GetattrRewriteArgs* rewrite_args) {
...
@@ -728,7 +728,7 @@ Box* Box::getattr(BoxedString* attr, GetattrRewriteArgs* rewrite_args) {
assert
(
hcls
->
type
==
HiddenClass
::
NORMAL
||
hcls
->
type
==
HiddenClass
::
SINGLETON
);
assert
(
hcls
->
type
==
HiddenClass
::
NORMAL
||
hcls
->
type
==
HiddenClass
::
SINGLETON
);
if
(
rewrite_args
)
{
if
(
unlikely
(
rewrite_args
)
)
{
if
(
!
rewrite_args
->
obj_hcls_guarded
)
{
if
(
!
rewrite_args
->
obj_hcls_guarded
)
{
if
(
cls
->
attrs_offset
<
0
)
{
if
(
cls
->
attrs_offset
<
0
)
{
REWRITE_ABORTED
(
""
);
REWRITE_ABORTED
(
""
);
...
@@ -1001,9 +1001,20 @@ Box* typeLookup(BoxedClass* cls, BoxedString* attr, GetattrRewriteArgs* rewrite_
...
@@ -1001,9 +1001,20 @@ Box* typeLookup(BoxedClass* cls, BoxedString* attr, GetattrRewriteArgs* rewrite_
return
NULL
;
return
NULL
;
}
else
{
}
else
{
assert
(
attr
->
interned_state
!=
SSTATE_NOT_INTERNED
);
assert
(
cls
->
tp_mro
);
assert
(
cls
->
tp_mro
);
assert
(
cls
->
tp_mro
->
cls
==
tuple_cls
);
assert
(
cls
->
tp_mro
->
cls
==
tuple_cls
);
for
(
auto
b
:
*
static_cast
<
BoxedTuple
*>
(
cls
->
tp_mro
))
{
for
(
auto
b
:
*
static_cast
<
BoxedTuple
*>
(
cls
->
tp_mro
))
{
// object_cls will get checked very often, but it only
// has attributes that start with an underscore.
if
(
b
==
object_cls
)
{
if
(
attr
->
data
()[
0
]
!=
'_'
)
{
assert
(
!
b
->
getattr
(
attr
,
NULL
));
continue
;
}
}
val
=
b
->
getattr
(
attr
,
NULL
);
val
=
b
->
getattr
(
attr
,
NULL
);
if
(
val
)
if
(
val
)
return
val
;
return
val
;
...
@@ -1674,7 +1685,7 @@ Box* getattrInternalGeneric(Box* obj, BoxedString* attr, GetattrRewriteArgs* rew
...
@@ -1674,7 +1685,7 @@ Box* getattrInternalGeneric(Box* obj, BoxedString* attr, GetattrRewriteArgs* rew
}
}
if
(
!
cls_only
)
{
if
(
!
cls_only
)
{
if
(
!
isSubclass
(
obj
->
cls
,
type_cls
))
{
if
(
!
PyType_Check
(
obj
))
{
// Look up the val in the object's dictionary and if you find it, return it.
// Look up the val in the object's dictionary and if you find it, return it.
Box
*
val
;
Box
*
val
;
...
...
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