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
452204c0
Commit
452204c0
authored
Jul 20, 2015
by
Kevin Modzelewski
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Reduce malloc() calls in rewriter
Allocating things in-line, using malloc-friendly data structures, etc.
parent
fa137eb6
Changes
4
Hide whitespace changes
Inline
Side-by-side
Showing
4 changed files
with
56 additions
and
44 deletions
+56
-44
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
No files found.
src/asm_writing/icinfo.cpp
View file @
452204c0
...
@@ -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 @
452204c0
...
@@ -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 @
452204c0
...
@@ -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,11 +218,11 @@ assembler::Register Rewriter::ConstLoader::loadConst(uint64_t val, Location othe
...
@@ -216,11 +218,11 @@ 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
))
return
reg
;
return
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 @
452204c0
...
@@ -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
();
...
...
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