Skip to content
Projeler
Gruplar
Parçacıklar
Yardım
Yükleniyor...
Oturum aç / Kaydol
Gezinmeyi değiştir
C
cpython
Proje
Proje
Ayrıntılar
Etkinlik
Cycle Analytics
Depo (repository)
Depo (repository)
Dosyalar
Kayıtlar (commit)
Dallar (branch)
Etiketler
Katkıda bulunanlar
Grafik
Karşılaştır
Grafikler
Konular (issue)
0
Konular (issue)
0
Liste
Pano
Etiketler
Kilometre Taşları
Birleştirme (merge) Talepleri
0
Birleştirme (merge) Talepleri
0
CI / CD
CI / CD
İş akışları (pipeline)
İşler
Zamanlamalar
Grafikler
Paketler
Paketler
Wiki
Wiki
Parçacıklar
Parçacıklar
Üyeler
Üyeler
Collapse sidebar
Close sidebar
Etkinlik
Grafik
Grafikler
Yeni bir konu (issue) oluştur
İşler
Kayıtlar (commit)
Konu (issue) Panoları
Kenar çubuğunu aç
Batuhan Osman TASKAYA
cpython
Commits
5e62ca4f
Kaydet (Commit)
5e62ca4f
authored
Tem 16, 2008
tarafından
Jesse Noller
Dosyalara gözat
Seçenekler
Dosyalara Gözat
İndir
Eposta Yamaları
Sade Fark
Apply patch for 874900: threading module can deadlock after fork
üst
1bbf4ea5
Show whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
130 additions
and
1 deletion
+130
-1
test_threading.py
Lib/test/test_threading.py
+79
-1
threading.py
Lib/threading.py
+31
-0
ceval.c
Python/ceval.c
+20
-0
No files found.
Lib/test/test_threading.py
Dosyayı görüntüle @
5e62ca4f
...
@@ -323,6 +323,82 @@ class ThreadTests(unittest.TestCase):
...
@@ -323,6 +323,82 @@ class ThreadTests(unittest.TestCase):
sys
.
getrefcount
(
weak_raising_cyclic_object
())))
sys
.
getrefcount
(
weak_raising_cyclic_object
())))
class
ThreadJoinOnShutdown
(
unittest
.
TestCase
):
def
_run_and_join
(
self
,
script
):
script
=
"""if 1:
import sys, os, time, threading
# a thread, which waits for the main program to terminate
def joiningfunc(mainthread):
mainthread.join()
print 'end of thread'
\n
"""
+
script
import
subprocess
p
=
subprocess
.
Popen
([
sys
.
executable
,
"-c"
,
script
],
stdout
=
subprocess
.
PIPE
)
rc
=
p
.
wait
()
self
.
assertEqual
(
p
.
stdout
.
read
(),
"end of main
\n
end of thread
\n
"
)
self
.
failIf
(
rc
==
2
,
"interpreter was blocked"
)
self
.
failUnless
(
rc
==
0
,
"Unexpected error"
)
def
test_1_join_on_shutdown
(
self
):
# The usual case: on exit, wait for a non-daemon thread
script
=
"""if 1:
import os
t = threading.Thread(target=joiningfunc,
args=(threading.current_thread(),))
t.start()
time.sleep(0.1)
print 'end of main'
"""
self
.
_run_and_join
(
script
)
def
test_2_join_in_forked_process
(
self
):
# Like the test above, but from a forked interpreter
import
os
if
not
hasattr
(
os
,
'fork'
):
return
script
=
"""if 1:
childpid = os.fork()
if childpid != 0:
os.waitpid(childpid, 0)
sys.exit(0)
t = threading.Thread(target=joiningfunc,
args=(threading.current_thread(),))
t.start()
print 'end of main'
"""
self
.
_run_and_join
(
script
)
def
test_3_join_in_forked_from_thread
(
self
):
# Like the test above, but fork() was called from a worker thread
# In the forked process, the main Thread object must be marked as stopped.
import
os
if
not
hasattr
(
os
,
'fork'
):
return
script
=
"""if 1:
main_thread = threading.current_thread()
def worker():
childpid = os.fork()
if childpid != 0:
os.waitpid(childpid, 0)
sys.exit(0)
t = threading.Thread(target=joiningfunc,
args=(main_thread,))
print 'end of main'
t.start()
t.join() # Should not block: main_thread is already stopped
w = threading.Thread(target=worker)
w.start()
"""
self
.
_run_and_join
(
script
)
class
ThreadingExceptionTests
(
unittest
.
TestCase
):
class
ThreadingExceptionTests
(
unittest
.
TestCase
):
# A RuntimeError should be raised if Thread.start() is called
# A RuntimeError should be raised if Thread.start() is called
# multiple times.
# multiple times.
...
@@ -363,7 +439,9 @@ class ThreadingExceptionTests(unittest.TestCase):
...
@@ -363,7 +439,9 @@ class ThreadingExceptionTests(unittest.TestCase):
def
test_main
():
def
test_main
():
test
.
test_support
.
run_unittest
(
ThreadTests
,
test
.
test_support
.
run_unittest
(
ThreadTests
,
ThreadingExceptionTests
)
ThreadJoinOnShutdown
,
ThreadingExceptionTests
,
)
if
__name__
==
"__main__"
:
if
__name__
==
"__main__"
:
test_main
()
test_main
()
Lib/threading.py
Dosyayı görüntüle @
5e62ca4f
...
@@ -825,6 +825,37 @@ except ImportError:
...
@@ -825,6 +825,37 @@ except ImportError:
from
_threading_local
import
local
from
_threading_local
import
local
def
_after_fork
():
# This function is called by Python/ceval.c:PyEval_ReInitThreads which
# is called from PyOS_AfterFork. Here we cleanup threading module state
# that should not exist after a fork.
# Reset _active_limbo_lock, in case we forked while the lock was held
# by another (non-forked) thread. http://bugs.python.org/issue874900
global
_active_limbo_lock
_active_limbo_lock
=
_allocate_lock
()
# fork() only copied the current thread; clear references to others.
new_active
=
{}
current
=
current_thread
()
with
_active_limbo_lock
:
for
ident
,
thread
in
_active
.
iteritems
():
if
thread
is
current
:
# There is only one active thread.
new_active
[
ident
]
=
thread
else
:
# All the others are already stopped.
# We don't call _Thread__stop() because it tries to acquire
# thread._Thread__block which could also have been held while
# we forked.
thread
.
_Thread__stopped
=
True
_limbo
.
clear
()
_active
.
clear
()
_active
.
update
(
new_active
)
assert
len
(
_active
)
==
1
# Self-test code
# Self-test code
def
_test
():
def
_test
():
...
...
Python/ceval.c
Dosyayı görüntüle @
5e62ca4f
...
@@ -274,6 +274,9 @@ PyEval_ReleaseThread(PyThreadState *tstate)
...
@@ -274,6 +274,9 @@ PyEval_ReleaseThread(PyThreadState *tstate)
void
void
PyEval_ReInitThreads
(
void
)
PyEval_ReInitThreads
(
void
)
{
{
PyObject
*
threading
,
*
result
;
PyThreadState
*
tstate
;
if
(
!
interpreter_lock
)
if
(
!
interpreter_lock
)
return
;
return
;
/*XXX Can't use PyThread_free_lock here because it does too
/*XXX Can't use PyThread_free_lock here because it does too
...
@@ -283,6 +286,23 @@ PyEval_ReInitThreads(void)
...
@@ -283,6 +286,23 @@ PyEval_ReInitThreads(void)
interpreter_lock
=
PyThread_allocate_lock
();
interpreter_lock
=
PyThread_allocate_lock
();
PyThread_acquire_lock
(
interpreter_lock
,
1
);
PyThread_acquire_lock
(
interpreter_lock
,
1
);
main_thread
=
PyThread_get_thread_ident
();
main_thread
=
PyThread_get_thread_ident
();
/* Update the threading module with the new state.
*/
tstate
=
PyThreadState_GET
();
threading
=
PyMapping_GetItemString
(
tstate
->
interp
->
modules
,
"threading"
);
if
(
threading
==
NULL
)
{
/* threading not imported */
PyErr_Clear
();
return
;
}
result
=
PyObject_CallMethod
(
threading
,
"_after_fork"
,
NULL
);
if
(
result
==
NULL
)
PyErr_WriteUnraisable
(
threading
);
else
Py_DECREF
(
result
);
Py_DECREF
(
threading
);
}
}
#endif
#endif
...
...
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