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
a80bc8b9
Commit
a80bc8b9
authored
Feb 25, 2016
by
Kevin Modzelewski
Browse files
Options
Browse Files
Download
Plain Diff
Merge commit '
67e30700
' into refcounting
parents
5a1345ca
67e30700
Changes
6
Hide whitespace changes
Inline
Side-by-side
Showing
6 changed files
with
75 additions
and
65 deletions
+75
-65
src/codegen/baseline_jit.cpp
src/codegen/baseline_jit.cpp
+13
-10
src/codegen/baseline_jit.h
src/codegen/baseline_jit.h
+3
-3
src/runtime/ics.cpp
src/runtime/ics.cpp
+23
-38
src/runtime/ics.h
src/runtime/ics.h
+1
-13
src/runtime/traceback.cpp
src/runtime/traceback.cpp
+3
-1
test/tests/traceback_test2.py
test/tests/traceback_test2.py
+32
-0
No files found.
src/codegen/baseline_jit.cpp
View file @
a80bc8b9
...
@@ -53,17 +53,20 @@ const unsigned char eh_info[]
...
@@ -53,17 +53,20 @@ const unsigned char eh_info[]
static_assert
(
JitCodeBlock
::
num_stack_args
==
2
,
"have to update EH table!"
);
static_assert
(
JitCodeBlock
::
num_stack_args
==
2
,
"have to update EH table!"
);
static_assert
(
JitCodeBlock
::
scratch_size
==
256
,
"have to update EH table!"
);
static_assert
(
JitCodeBlock
::
scratch_size
==
256
,
"have to update EH table!"
);
constexpr
int
code_size
=
JitCodeBlock
::
memory_size
-
sizeof
(
eh_info
);
JitCodeBlock
::
JitCodeBlock
(
llvm
::
StringRef
name
)
JitCodeBlock
::
JitCodeBlock
(
llvm
::
StringRef
name
)
:
code
(
new
uint8_t
[
code_size
]),
:
memory
(
new
uint8_t
[
memory_size
]),
eh_frame
(
new
uint8_t
[
sizeof
(
eh_info
)]),
entry_offset
(
0
),
entry_offset
(
0
),
a
(
code
.
get
(
),
code_size
),
a
(
memory
.
get
()
+
sizeof
(
eh_info
),
code_size
),
is_currently_writing
(
false
),
is_currently_writing
(
false
),
asm_failed
(
false
)
{
asm_failed
(
false
)
{
static
StatCounter
num_jit_code_blocks
(
"num_baselinejit_code_blocks"
);
static
StatCounter
num_jit_code_blocks
(
"num_baselinejit_code_blocks"
);
num_jit_code_blocks
.
log
();
num_jit_code_blocks
.
log
();
static
StatCounter
num_jit_total_bytes
(
"num_baselinejit_total_bytes"
);
static
StatCounter
num_jit_total_bytes
(
"num_baselinejit_total_bytes"
);
num_jit_total_bytes
.
log
(
code_size
);
num_jit_total_bytes
.
log
(
memory_size
);
uint8_t
*
code
=
a
.
curInstPointer
();
// emit prolog
// emit prolog
a
.
push
(
assembler
::
R14
);
a
.
push
(
assembler
::
R14
);
...
@@ -78,20 +81,20 @@ JitCodeBlock::JitCodeBlock(llvm::StringRef name)
...
@@ -78,20 +81,20 @@ JitCodeBlock::JitCodeBlock(llvm::StringRef name)
// generate the eh frame...
// generate the eh frame...
const
int
size
=
sizeof
(
eh_info
);
const
int
size
=
sizeof
(
eh_info
);
void
*
eh_frame_addr
=
eh_frame
.
get
();
void
*
eh_frame_addr
=
memory
.
get
();
memcpy
(
eh_frame_addr
,
eh_info
,
size
);
memcpy
(
eh_frame_addr
,
eh_info
,
size
);
int32_t
*
offset_ptr
=
(
int32_t
*
)((
uint8_t
*
)
eh_frame_addr
+
0x20
);
int32_t
*
offset_ptr
=
(
int32_t
*
)((
uint8_t
*
)
eh_frame_addr
+
0x20
);
int32_t
*
size_ptr
=
(
int32_t
*
)((
uint8_t
*
)
eh_frame_addr
+
0x24
);
int32_t
*
size_ptr
=
(
int32_t
*
)((
uint8_t
*
)
eh_frame_addr
+
0x24
);
int64_t
offset
=
(
int8_t
*
)
code
.
get
()
-
(
int8_t
*
)
offset_ptr
;
int64_t
offset
=
(
int8_t
*
)
code
-
(
int8_t
*
)
offset_ptr
;
RELEASE_ASSERT
(
offset
>=
INT_MIN
&&
offset
<=
INT_MAX
,
""
);
assert
(
offset
>=
INT_MIN
&&
offset
<=
INT_MAX
);
*
offset_ptr
=
offset
;
*
offset_ptr
=
offset
;
*
size_ptr
=
code_size
;
*
size_ptr
=
code_size
;
registerDynamicEhFrame
((
uint64_t
)
code
.
get
()
,
code_size
,
(
uint64_t
)
eh_frame_addr
,
size
-
4
);
registerDynamicEhFrame
((
uint64_t
)
code
,
code_size
,
(
uint64_t
)
eh_frame_addr
,
size
-
4
);
registerEHFrames
((
uint8_t
*
)
eh_frame_addr
,
(
uint64_t
)
eh_frame_addr
,
size
);
registerEHFrames
((
uint8_t
*
)
eh_frame_addr
,
(
uint64_t
)
eh_frame_addr
,
size
);
g
.
func_addr_registry
.
registerFunction
((
"bjit_"
+
name
).
str
(),
code
.
get
()
,
code_size
,
NULL
);
g
.
func_addr_registry
.
registerFunction
((
"bjit_"
+
name
).
str
(),
code
,
code_size
,
NULL
);
}
}
std
::
unique_ptr
<
JitFragmentWriter
>
JitCodeBlock
::
newFragment
(
CFGBlock
*
block
,
int
patch_jump_offset
)
{
std
::
unique_ptr
<
JitFragmentWriter
>
JitCodeBlock
::
newFragment
(
CFGBlock
*
block
,
int
patch_jump_offset
)
{
...
@@ -621,7 +624,7 @@ int JitFragmentWriter::finishCompilation() {
...
@@ -621,7 +624,7 @@ int JitFragmentWriter::finishCompilation() {
int
bytes_written
=
assembler
->
bytesWritten
();
int
bytes_written
=
assembler
->
bytesWritten
();
// don't retry JITing very large blocks
// don't retry JITing very large blocks
const
auto
large_block_threshold
=
JitCodeBlock
::
code_size
-
4096
;
const
auto
large_block_threshold
=
code_size
-
4096
;
if
(
bytes_written
>
large_block_threshold
)
{
if
(
bytes_written
>
large_block_threshold
)
{
static
StatCounter
num_jit_large_blocks
(
"num_baselinejit_skipped_large_blocks"
);
static
StatCounter
num_jit_large_blocks
(
"num_baselinejit_skipped_large_blocks"
);
num_jit_large_blocks
.
log
();
num_jit_large_blocks
.
log
();
...
...
src/codegen/baseline_jit.h
View file @
a80bc8b9
...
@@ -132,7 +132,7 @@ class JitFragmentWriter;
...
@@ -132,7 +132,7 @@ class JitFragmentWriter;
class
JitCodeBlock
{
class
JitCodeBlock
{
public:
public:
static
constexpr
int
scratch_size
=
256
;
static
constexpr
int
scratch_size
=
256
;
static
constexpr
int
code_size
=
32768
;
static
constexpr
int
memory_size
=
32768
;
// must fit the EH frame + generated code
static
constexpr
int
num_stack_args
=
2
;
static
constexpr
int
num_stack_args
=
2
;
// scratch size + space for passing additional args on the stack without having to adjust the SP when calling
// scratch size + space for passing additional args on the stack without having to adjust the SP when calling
...
@@ -140,8 +140,8 @@ public:
...
@@ -140,8 +140,8 @@ public:
static
constexpr
int
sp_adjustment
=
scratch_size
+
num_stack_args
*
8
+
8
/* = alignment */
;
static
constexpr
int
sp_adjustment
=
scratch_size
+
num_stack_args
*
8
+
8
/* = alignment */
;
private:
private:
std
::
unique_ptr
<
uint8_t
[]
>
code
;
// the memory block contains the EH frame directly followed by the generated machine code.
std
::
unique_ptr
<
uint8_t
[]
>
eh_frame
;
std
::
unique_ptr
<
uint8_t
[]
>
memory
;
int
entry_offset
;
int
entry_offset
;
assembler
::
Assembler
a
;
assembler
::
Assembler
a
;
bool
is_currently_writing
;
bool
is_currently_writing
;
...
...
src/runtime/ics.cpp
View file @
a80bc8b9
...
@@ -154,7 +154,12 @@ static const char _eh_frame_template_fp[] =
...
@@ -154,7 +154,12 @@ static const char _eh_frame_template_fp[] =
static
constexpr
int
_eh_frame_template_ofp_size
=
sizeof
(
_eh_frame_template_ofp
)
-
1
;
static
constexpr
int
_eh_frame_template_ofp_size
=
sizeof
(
_eh_frame_template_ofp
)
-
1
;
static
constexpr
int
_eh_frame_template_fp_size
=
sizeof
(
_eh_frame_template_fp
)
-
1
;
static
constexpr
int
_eh_frame_template_fp_size
=
sizeof
(
_eh_frame_template_fp
)
-
1
;
#define EH_FRAME_SIZE (sizeof(_eh_frame_template) - 1) // omit string-terminating null byte
#if RUNTIMEICS_OMIT_FRAME_PTR
#define EH_FRAME_SIZE _eh_frame_template_ofp_size
#else
#define EH_FRAME_SIZE _eh_frame_template_fp_size;
#endif
static_assert
(
sizeof
(
""
)
==
1
,
"strings are null-terminated"
);
static_assert
(
sizeof
(
""
)
==
1
,
"strings are null-terminated"
);
...
@@ -175,35 +180,6 @@ static void writeTrivialEhFrame(void* eh_frame_addr, void* func_addr, uint64_t f
...
@@ -175,35 +180,6 @@ static void writeTrivialEhFrame(void* eh_frame_addr, void* func_addr, uint64_t f
*
size_ptr
=
func_size
;
*
size_ptr
=
func_size
;
}
}
void
EHFrameManager
::
writeAndRegister
(
void
*
func_addr
,
uint64_t
func_size
)
{
assert
(
eh_frame_addr
==
NULL
);
const
int
size
=
omit_frame_pointer
?
_eh_frame_template_ofp_size
:
_eh_frame_template_fp_size
;
#ifdef NVALGRIND
eh_frame_addr
=
malloc
(
size
);
#else
eh_frame_addr
=
mmap
(
NULL
,
(
size
+
(
PAGE_SIZE
-
1
))
&
~
(
PAGE_SIZE
-
1
),
PROT_READ
|
PROT_WRITE
,
MAP_PRIVATE
|
MAP_ANONYMOUS
,
-
1
,
0
);
RELEASE_ASSERT
(
eh_frame_addr
!=
MAP_FAILED
,
""
);
#endif
writeTrivialEhFrame
(
eh_frame_addr
,
func_addr
,
func_size
,
omit_frame_pointer
);
// (EH_FRAME_SIZE - 4) to omit the 4-byte null terminator, otherwise we trip an assert in parseEhFrame.
// TODO: can we omit the terminator in general?
registerDynamicEhFrame
((
uint64_t
)
func_addr
,
func_size
,
(
uint64_t
)
eh_frame_addr
,
size
-
4
);
registerEHFrames
((
uint8_t
*
)
eh_frame_addr
,
(
uint64_t
)
eh_frame_addr
,
size
);
}
EHFrameManager
::~
EHFrameManager
()
{
if
(
eh_frame_addr
)
{
const
int
size
=
omit_frame_pointer
?
_eh_frame_template_ofp_size
:
_eh_frame_template_fp_size
;
deregisterEHFrames
((
uint8_t
*
)
eh_frame_addr
,
(
uint64_t
)
eh_frame_addr
,
size
);
#ifdef NVALGRIND
free
(
eh_frame_addr
);
#else
munmap
(
eh_frame_addr
,
(
size
+
(
PAGE_SIZE
-
1
))
&
~
(
PAGE_SIZE
-
1
));
#endif
}
}
#if RUNTIMEICS_OMIT_FRAME_PTR
#if RUNTIMEICS_OMIT_FRAME_PTR
// If you change this, you *must* update the value in _eh_frame_template
// If you change this, you *must* update the value in _eh_frame_template
// (set the -9'th byte to this value plus 8)
// (set the -9'th byte to this value plus 8)
...
@@ -212,7 +188,7 @@ EHFrameManager::~EHFrameManager() {
...
@@ -212,7 +188,7 @@ EHFrameManager::~EHFrameManager() {
#define SCRATCH_BYTES 0x30
#define SCRATCH_BYTES 0x30
#endif
#endif
RuntimeIC
::
RuntimeIC
(
void
*
func_addr
,
int
num_slots
,
int
slot_size
)
:
eh_frame
(
RUNTIMEICS_OMIT_FRAME_PTR
)
{
RuntimeIC
::
RuntimeIC
(
void
*
func_addr
,
int
num_slots
,
int
slot_size
)
{
static
StatCounter
sc
(
"runtime_ics_num"
);
static
StatCounter
sc
(
"runtime_ics_num"
);
sc
.
log
();
sc
.
log
();
...
@@ -254,15 +230,20 @@ RuntimeIC::RuntimeIC(void* func_addr, int num_slots, int slot_size) : eh_frame(R
...
@@ -254,15 +230,20 @@ RuntimeIC::RuntimeIC(void* func_addr, int num_slots, int slot_size) : eh_frame(R
int
patchable_size
=
num_slots
*
slot_size
;
int
patchable_size
=
num_slots
*
slot_size
;
int
total_code_size
=
PROLOGUE_SIZE
+
patchable_size
+
CALL_SIZE
+
EPILOGUE_SIZE
;
#ifdef NVALGRIND
#ifdef NVALGRIND
int
total_size
=
PROLOGUE_SIZE
+
patchable_size
+
CALL_SIZE
+
EPILOGU
E_SIZE
;
int
total_size
=
total_code_size
+
EH_FRAM
E_SIZE
;
addr
=
malloc
(
total_size
);
addr
=
malloc
(
total_size
);
#else
#else
total_size
=
PROLOGUE_SIZE
+
patchable_size
+
CALL_SIZE
+
EPILOGU
E_SIZE
;
total_size
=
total_code_size
+
EH_FRAM
E_SIZE
;
addr
=
mmap
(
NULL
,
(
total_size
+
(
PAGE_SIZE
-
1
))
&
~
(
PAGE_SIZE
-
1
),
PROT_READ
|
PROT_WRITE
|
PROT_EXEC
,
addr
=
mmap
(
NULL
,
(
total_size
+
(
PAGE_SIZE
-
1
))
&
~
(
PAGE_SIZE
-
1
),
PROT_READ
|
PROT_WRITE
|
PROT_EXEC
,
MAP_PRIVATE
|
MAP_ANONYMOUS
,
-
1
,
0
);
MAP_PRIVATE
|
MAP_ANONYMOUS
,
-
1
,
0
);
RELEASE_ASSERT
(
addr
!=
MAP_FAILED
,
""
);
RELEASE_ASSERT
(
addr
!=
MAP_FAILED
,
""
);
#endif
#endif
// the memory block contains the EH frame directly followed by the generated machine code.
void
*
eh_frame_addr
=
addr
;
addr
=
(
char
*
)
addr
+
EH_FRAME_SIZE
;
// printf("Allocated runtime IC at %p\n", addr);
// printf("Allocated runtime IC at %p\n", addr);
...
@@ -308,9 +289,11 @@ RuntimeIC::RuntimeIC(void* func_addr, int num_slots, int slot_size) : eh_frame(R
...
@@ -308,9 +289,11 @@ RuntimeIC::RuntimeIC(void* func_addr, int num_slots, int slot_size) : eh_frame(R
assert
(
!
epilogue_assem
.
hasFailed
());
assert
(
!
epilogue_assem
.
hasFailed
());
assert
(
epilogue_assem
.
isExactlyFull
());
assert
(
epilogue_assem
.
isExactlyFull
());
// TODO: ideally would be more intelligent about allocation strategies.
writeTrivialEhFrame
(
eh_frame_addr
,
addr
,
total_code_size
,
RUNTIMEICS_OMIT_FRAME_PTR
);
// The code sections should be together and the eh sections together
// (EH_FRAME_SIZE - 4) to omit the 4-byte null terminator, otherwise we trip an assert in parseEhFrame.
eh_frame
.
writeAndRegister
(
addr
,
total_size
);
// TODO: can we omit the terminator in general?
registerDynamicEhFrame
((
uint64_t
)
addr
,
total_code_size
,
(
uint64_t
)
eh_frame_addr
,
EH_FRAME_SIZE
-
4
);
registerEHFrames
((
uint8_t
*
)
eh_frame_addr
,
(
uint64_t
)
eh_frame_addr
,
EH_FRAME_SIZE
);
}
else
{
}
else
{
addr
=
func_addr
;
addr
=
func_addr
;
}
}
...
@@ -319,10 +302,12 @@ RuntimeIC::RuntimeIC(void* func_addr, int num_slots, int slot_size) : eh_frame(R
...
@@ -319,10 +302,12 @@ RuntimeIC::RuntimeIC(void* func_addr, int num_slots, int slot_size) : eh_frame(R
RuntimeIC
::~
RuntimeIC
()
{
RuntimeIC
::~
RuntimeIC
()
{
if
(
ENABLE_RUNTIME_ICS
)
{
if
(
ENABLE_RUNTIME_ICS
)
{
deregisterCompiledPatchpoint
(
icinfo
.
get
());
deregisterCompiledPatchpoint
(
icinfo
.
get
());
uint8_t
*
eh_frame_addr
=
(
uint8_t
*
)
addr
-
EH_FRAME_SIZE
;
deregisterEHFrames
(
eh_frame_addr
,
(
uint64_t
)
eh_frame_addr
,
EH_FRAME_SIZE
);
#ifdef NVALGRIND
#ifdef NVALGRIND
free
(
addr
);
free
(
eh_frame_
addr
);
#else
#else
munmap
(
addr
,
total_size
);
munmap
(
eh_frame_
addr
,
total_size
);
#endif
#endif
}
else
{
}
else
{
}
}
...
...
src/runtime/ics.h
View file @
a80bc8b9
...
@@ -22,24 +22,12 @@ namespace pyston {
...
@@ -22,24 +22,12 @@ namespace pyston {
class
ICInfo
;
class
ICInfo
;
class
EHFrameManager
{
private:
void
*
eh_frame_addr
;
bool
omit_frame_pointer
;
public:
EHFrameManager
(
bool
omit_frame_pointer
)
:
eh_frame_addr
(
NULL
),
omit_frame_pointer
(
omit_frame_pointer
)
{}
~
EHFrameManager
();
void
writeAndRegister
(
void
*
func_addr
,
uint64_t
func_size
);
};
class
RuntimeIC
{
class
RuntimeIC
{
private:
private:
void
*
addr
;
void
*
addr
;
// points to function start not the start of the allocated memory block.
#ifndef NVALGRIND
#ifndef NVALGRIND
size_t
total_size
;
size_t
total_size
;
#endif
#endif
EHFrameManager
eh_frame
;
std
::
unique_ptr
<
ICInfo
>
icinfo
;
std
::
unique_ptr
<
ICInfo
>
icinfo
;
...
...
src/runtime/traceback.cpp
View file @
a80bc8b9
...
@@ -159,7 +159,9 @@ void setupTraceback() {
...
@@ -159,7 +159,9 @@ void setupTraceback() {
NULL));
NULL));
*/
*/
traceback_cls
->
giveAttrDescriptor
(
"tb_next"
,
traceback_tb_next
,
NULL
);
traceback_cls
->
giveAttrDescriptor
(
"tb_next"
,
traceback_tb_next
,
NULL
);
traceback_cls
->
giveAttr
(
"tb_lineno"
,
new
BoxedMemberDescriptor
(
BoxedMemberDescriptor
::
INT
,
offsetof
(
BoxedTraceback
,
line
)
+
offsetof
(
LineInfo
,
line
)));
traceback_cls
->
freeze
();
traceback_cls
->
freeze
();
}
}
}
}
test/tests/traceback_test2.py
0 → 100644
View file @
a80bc8b9
import
sys
,
traceback
def
lumberjack
():
bright_side_of_death
()
def
bright_side_of_death
():
return
tuple
()[
0
]
try
:
lumberjack
()
except
IndexError
:
exc_type
,
exc_value
,
exc_traceback
=
sys
.
exc_info
()
print
"*** print_tb:"
traceback
.
print_tb
(
exc_traceback
,
limit
=
1
,
file
=
sys
.
stdout
)
print
"*** print_exception:"
traceback
.
print_exception
(
exc_type
,
exc_value
,
exc_traceback
,
limit
=
2
,
file
=
sys
.
stdout
)
print
"*** print_exc:"
traceback
.
print_exc
()
print
"*** format_exc, first and last line:"
formatted_lines
=
traceback
.
format_exc
().
splitlines
()
print
formatted_lines
[
0
]
print
formatted_lines
[
-
1
]
print
"*** format_exception:"
print
repr
(
traceback
.
format_exception
(
exc_type
,
exc_value
,
exc_traceback
))
print
"*** extract_tb:"
print
repr
(
traceback
.
extract_tb
(
exc_traceback
))
print
"*** format_tb:"
print
repr
(
traceback
.
format_tb
(
exc_traceback
))
print
"*** tb_lineno:"
,
exc_traceback
.
tb_lineno
assert
exc_traceback
.
tb_lineno
==
10
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