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
ce62f37e
Commit
ce62f37e
authored
Jul 28, 2015
by
Kevin Modzelewski
Browse files
Options
Browse Files
Download
Plain Diff
Merge pull request #771 from kmod/threading_local_debugging
Add some PerThreadSet features
parents
f49d0937
273bf3d8
Changes
6
Hide whitespace changes
Inline
Side-by-side
Showing
6 changed files
with
96 additions
and
7 deletions
+96
-7
src/core/thread_utils.h
src/core/thread_utils.h
+51
-3
src/core/threading.cpp
src/core/threading.cpp
+3
-1
src/gc/collector.cpp
src/gc/collector.cpp
+3
-3
src/gc/heap.cpp
src/gc/heap.cpp
+30
-0
src/gc/heap.h
src/gc/heap.h
+6
-0
test/tests/multiprocessing_test.py
test/tests/multiprocessing_test.py
+3
-0
No files found.
src/core/thread_utils.h
View file @
ce62f37e
...
...
@@ -144,7 +144,28 @@ template <int N, int... S> struct gens : gens<N - 1, N - 1, S...> {};
template
<
int
...
S
>
struct
gens
<
0
,
S
...
>
{
typedef
seq
<
S
...
>
type
;
};
}
template
<
typename
T
,
typename
...
CtorArgs
>
class
PerThreadSet
{
class
PerThreadSetBase
{
private:
static
std
::
unordered_set
<
PerThreadSetBase
*>
all_instances
;
protected:
PerThreadSetBase
()
{
all_instances
.
insert
(
this
);
}
virtual
~
PerThreadSetBase
()
{
assert
(
all_instances
.
count
(
this
));
all_instances
.
erase
(
this
);
}
virtual
void
onFork
()
=
0
;
public:
static
void
runAllForkHandlers
()
{
for
(
auto
inst
:
all_instances
)
inst
->
onFork
();
}
};
template
<
typename
T
,
typename
...
CtorArgs
>
class
PerThreadSet
:
public
PerThreadSetBase
{
private:
pthread_key_t
pthread_key
;
PthreadFastMutex
lock
;
...
...
@@ -159,20 +180,26 @@ private:
std
::
unordered_map
<
pthread_t
,
Storage
*>
map
;
std
::
tuple
<
CtorArgs
...
>
ctor_args
;
#ifndef NDEBUG
int
map_elts
=
0
;
#endif
static
void
dtor
(
void
*
val
)
{
Storage
*
s
=
static_cast
<
Storage
*>
(
val
);
assert
(
s
);
auto
*
self
=
s
->
self
;
LOCK_REGION
(
&
self
->
lock
);
ASSERT
(
self
->
map
.
size
()
==
self
->
map_elts
,
"%ld %d"
,
self
->
map
.
size
(),
self
->
map_elts
);
assert
(
s
->
my_tid
==
pthread_self
());
// I assume this destructor gets called on the same thread
// that this data is bound to:
assert
(
self
->
map
.
count
(
pthread_self
()));
self
->
map
.
erase
(
pthread_self
());
#ifndef NDEBUG
self
->
map_elts
--
;
#endif
delete
s
;
}
...
...
@@ -185,6 +212,21 @@ private:
.
val
=
T
(
std
::
get
<
S
>
(
ctor_args
)...)
};
}
protected:
void
onFork
()
override
{
pthread_t
surviving_ptid
=
pthread_self
();
for
(
auto
it
=
this
->
map
.
begin
(),
end
=
this
->
map
.
end
();
it
!=
end
;)
{
if
(
it
->
first
!=
surviving_ptid
)
{
delete
it
->
second
;
it
=
this
->
map
.
erase
(
it
);
#ifndef NDEBUG
this
->
map_elts
--
;
#endif
}
else
++
it
;
}
}
public:
PerThreadSet
(
CtorArgs
...
ctor_args
)
:
ctor_args
(
std
::
forward
<
CtorArgs
>
(
ctor_args
)...)
{
int
code
=
pthread_key_create
(
&
pthread_key
,
&
dtor
);
...
...
@@ -214,6 +256,12 @@ public:
s
=
make
(
typename
impl
::
gens
<
sizeof
...(
CtorArgs
)
>::
type
());
LOCK_REGION
(
&
lock
);
#ifndef NDEBUG
assert
(
map
.
size
()
==
map_elts
);
map_elts
++
;
#endif
int
code
=
pthread_setspecific
(
pthread_key
,
s
);
assert
(
code
==
0
);
...
...
src/core/threading.cpp
View file @
ce62f37e
...
...
@@ -35,6 +35,8 @@
namespace
pyston
{
namespace
threading
{
std
::
unordered_set
<
PerThreadSetBase
*>
PerThreadSetBase
::
all_instances
;
extern
"C"
{
__thread
PyThreadState
cur_thread_state
=
{
0
,
NULL
,
NULL
,
NULL
,
NULL
};
// not sure if we need to explicitly request zero-initialization
...
...
@@ -508,7 +510,7 @@ extern "C" void PyEval_ReInitThreads() noexcept {
num_starting_threads
=
0
;
threads_waiting_on_gil
=
0
;
// TODO we should clean up all created PerThreadSets, such as the one used in the heap for thread-local-caches.
PerThreadSetBase
::
runAllForkHandlers
();
}
void
acquireGLWrite
()
{
...
...
src/gc/collector.cpp
View file @
ce62f37e
...
...
@@ -113,10 +113,10 @@ private:
public:
TraceStack
(
TraceStackType
type
)
:
visit_type
(
type
)
{
get_chunk
();
}
TraceStack
(
TraceStackType
type
,
const
std
::
unordered_set
<
void
*>&
root
_handle
s
)
:
visit_type
(
type
)
{
TraceStack
(
TraceStackType
type
,
const
std
::
unordered_set
<
void
*>&
roots
)
:
visit_type
(
type
)
{
get_chunk
();
for
(
void
*
p
:
root
_handle
s
)
{
assert
(
!
isMarked
(
GCAllocation
::
fromUserData
(
p
))
);
for
(
void
*
p
:
roots
)
{
ASSERT
(
!
isMarked
(
GCAllocation
::
fromUserData
(
p
)),
""
);
push
(
p
);
}
}
...
...
src/gc/heap.cpp
View file @
ce62f37e
...
...
@@ -411,7 +411,37 @@ GCAllocation* SmallArena::allocationFrom(void* ptr) {
return
reinterpret_cast
<
GCAllocation
*>
(
&
b
->
atoms
[
atom_idx
]);
}
#ifndef NDEBUG
void
SmallArena
::
assertConsistent
()
{
std
::
unordered_set
<
Block
*>
seen_blocks
;
auto
scan
=
[
&
seen_blocks
](
Block
*
h
)
{
while
(
h
)
{
ASSERT
(
h
>=
(
void
*
)
SMALL_ARENA_START
&&
h
<
(
void
*
)
LARGE_ARENA_START
,
"%p"
,
h
);
assert
(
!
seen_blocks
.
count
(
h
));
seen_blocks
.
insert
(
h
);
if
(
h
->
next
)
assert
(
h
->
next
->
prev
==
&
h
->
next
);
h
=
h
->
next
;
}
};
thread_caches
.
forEachValue
([
&
scan
](
ThreadBlockCache
*
cache
)
{
for
(
int
bidx
=
0
;
bidx
<
NUM_BUCKETS
;
bidx
++
)
{
scan
(
cache
->
cache_free_heads
[
bidx
]);
scan
(
cache
->
cache_full_heads
[
bidx
]);
}
});
for
(
int
bidx
=
0
;
bidx
<
NUM_BUCKETS
;
bidx
++
)
{
scan
(
full_heads
[
bidx
]);
scan
(
heads
[
bidx
]);
}
}
#endif
void
SmallArena
::
freeUnmarked
(
std
::
vector
<
Box
*>&
weakly_referenced
)
{
assertConsistent
();
thread_caches
.
forEachValue
([
this
,
&
weakly_referenced
](
ThreadBlockCache
*
cache
)
{
for
(
int
bidx
=
0
;
bidx
<
NUM_BUCKETS
;
bidx
++
)
{
Block
*
h
=
cache
->
cache_free_heads
[
bidx
];
...
...
src/gc/heap.h
View file @
ce62f37e
...
...
@@ -274,6 +274,12 @@ public:
void
prepareForCollection
()
{}
void
cleanupAfterCollection
()
{}
#ifndef NDEBUG
void
assertConsistent
();
#else
void
assertConsistent
()
{}
#endif
private:
template
<
int
N
>
class
Bitmap
{
static_assert
(
N
%
64
==
0
,
""
);
...
...
test/tests/multiprocessing_test.py
View file @
ce62f37e
...
...
@@ -17,6 +17,7 @@ if __name__ == '__main__':
p
=
multiprocessing
.
Process
(
target
=
f
,
args
=
(
'bob'
,))
p
.
start
()
p
.
join
()
print
p
.
exitcode
def
f
(
q
):
...
...
@@ -28,3 +29,5 @@ if __name__ == '__main__':
p
.
start
()
print
q
.
get
()
# prints "[42, None, 'hello']"
p
.
join
()
print
p
.
exitcode
print
"done"
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