Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
G
gevent
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
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Kirill Smelkov
gevent
Commits
fd57ead7
Commit
fd57ead7
authored
Jun 23, 2015
by
Jason Madden
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Fix a race condition stopping watchers from the new callback. Remove some unused code.
parent
aebf5c51
Changes
1
Hide whitespace changes
Inline
Side-by-side
Showing
1 changed file
with
25 additions
and
24 deletions
+25
-24
gevent/corecffi.py
gevent/corecffi.py
+25
-24
No files found.
gevent/corecffi.py
View file @
fd57ead7
...
@@ -291,25 +291,39 @@ def _python_callback(handle, revents):
...
@@ -291,25 +291,39 @@ def _python_callback(handle, revents):
return
0
return
0
libev
.
python_callback
=
_python_callback
libev
.
python_callback
=
_python_callback
# After _python_callback is called, the handle may no longer be
# valid. The callback itself might have called watcher.stop(),
# which would remove the object from loop.keepaliveset, and if
# that was the last reference to it, the handle would be GC'd.
# Therefore the other functions need to correctly deal with an
# invalid handle
@
ffi
.
callback
(
"void(void* handle, int revents)"
)
@
ffi
.
callback
(
"void(void* handle, int revents)"
)
def
_python_handle_error
(
handle
,
revents
):
def
_python_handle_error
(
handle
,
revents
):
watcher
=
ffi
.
from_handle
(
handle
)
try
:
watcher
=
ffi
.
from_handle
(
handle
)
except
RuntimeError
:
return
exc_info
=
watcher
.
_exc_info
exc_info
=
watcher
.
_exc_info
del
watcher
.
_exc_info
del
watcher
.
_exc_info
try
:
try
:
watcher
.
loop
.
handle_error
(
watcher
,
*
exc_info
)
watcher
.
loop
.
handle_error
(
watcher
,
*
exc_info
)
finally
:
finally
:
if
revents
&
(
libev
.
EV_READ
|
libev
.
EV_WRITE
):
if
revents
&
(
libev
.
EV_READ
|
libev
.
EV_WRITE
):
try
:
try
:
watcher
.
stop
()
watcher
.
stop
()
except
:
except
:
watcher
.
loop
.
handle_error
(
watcher
,
*
sys
.
exc_info
())
watcher
.
loop
.
handle_error
(
watcher
,
*
sys
.
exc_info
())
return
return
libev
.
python_handle_error
=
_python_handle_error
libev
.
python_handle_error
=
_python_handle_error
@
ffi
.
callback
(
"void(void* handle)"
)
@
ffi
.
callback
(
"void(void* handle)"
)
def
_python_stop
(
handle
):
def
_python_stop
(
handle
):
watcher
=
ffi
.
from_handle
(
handle
)
try
:
watcher
=
ffi
.
from_handle
(
handle
)
except
RuntimeError
:
return
watcher
.
stop
()
watcher
.
stop
()
libev
.
python_stop
=
_python_stop
libev
.
python_stop
=
_python_stop
...
@@ -491,10 +505,14 @@ class loop(object):
...
@@ -491,10 +505,14 @@ class loop(object):
self
.
_in_callback
=
False
self
.
_in_callback
=
False
self
.
_callbacks
=
[]
self
.
_callbacks
=
[]
# self._check is a watcher that runs in each iteration of the
# mainloop, just after the blocking call
self
.
_check
=
ffi
.
new
(
"struct ev_check *"
)
self
.
_check
=
ffi
.
new
(
"struct ev_check *"
)
self
.
_check_callback_ffi
=
ffi
.
callback
(
"void(*)(struct ev_loop *, void*, int)"
,
self
.
_check_callback
)
self
.
_check_callback_ffi
=
ffi
.
callback
(
"void(*)(struct ev_loop *, void*, int)"
,
self
.
_check_callback
)
libev
.
ev_check_init
(
self
.
_check
,
self
.
_check_callback_ffi
)
libev
.
ev_check_init
(
self
.
_check
,
self
.
_check_callback_ffi
)
# self._prepare is a watcher that runs in each iteration of the mainloop,
# just before the blocking call
self
.
_prepare
=
ffi
.
new
(
"struct ev_prepare *"
)
self
.
_prepare
=
ffi
.
new
(
"struct ev_prepare *"
)
self
.
_prepare_callback_ffi
=
ffi
.
callback
(
"void(*)(struct ev_loop *, void*, int)"
,
self
.
_run_callbacks
)
self
.
_prepare_callback_ffi
=
ffi
.
callback
(
"void(*)(struct ev_loop *, void*, int)"
,
self
.
_run_callbacks
)
libev
.
ev_prepare_init
(
self
.
_prepare
,
self
.
_prepare_callback_ffi
)
libev
.
ev_prepare_init
(
self
.
_prepare
,
self
.
_prepare_callback_ffi
)
...
@@ -542,17 +560,6 @@ class loop(object):
...
@@ -542,17 +560,6 @@ class loop(object):
raise
KeyboardInterrupt
raise
KeyboardInterrupt
self
.
ate_keyboard_interrupt
=
True
self
.
ate_keyboard_interrupt
=
True
def
_wrap_cb
(
self
,
cb
):
def
wrapper
(
*
args
):
try
:
self
.
keyboard_interrupt_allowed
=
True
return
cb
(
*
args
)
except
:
self
.
handle_error
((
cb
,
args
),
*
sys
.
exc_info
())
finally
:
self
.
keyboard_interrupt_allowed
=
False
return
wrapper
def
_run_callbacks
(
self
,
evloop
,
_
,
revents
):
def
_run_callbacks
(
self
,
evloop
,
_
,
revents
):
count
=
1000
count
=
1000
libev
.
ev_timer_stop
(
self
.
_ptr
,
self
.
_timer0
)
libev
.
ev_timer_stop
(
self
.
_ptr
,
self
.
_timer0
)
...
@@ -1136,12 +1143,6 @@ class stat(watcher):
...
@@ -1136,12 +1143,6 @@ class stat(watcher):
args
=
(
ffi
.
new
(
'char[]'
,
path
),
args
=
(
ffi
.
new
(
'char[]'
,
path
),
interval
))
interval
))
def
start
(
self
,
callback
,
*
args
):
watcher
.
start
(
self
,
callback
,
*
args
)
def
stop
(
self
):
watcher
.
stop
(
self
)
@
property
@
property
def
path
(
self
):
def
path
(
self
):
return
ffi
.
string
(
self
.
_watcher
.
path
)
return
ffi
.
string
(
self
.
_watcher
.
path
)
...
...
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