Skip to content
Projects
Groups
Snippets
Help
Loading...
Sign in / Register
Toggle navigation
L
libzmq
Project
Project
Details
Activity
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Packages
Packages
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
submodule
libzmq
Commits
55886b8b
Commit
55886b8b
authored
Feb 13, 2015
by
Pieter Hintjens
Browse files
Options
Browse Files
Download
Plain Diff
Merge pull request #1352 from somdoron/master
thread safety bug - lock when sending
parents
cac4d7aa
deaa8965
Hide whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
22 additions
and
28 deletions
+22
-28
mailbox_safe.cpp
src/mailbox_safe.cpp
+8
-19
mailbox_safe.hpp
src/mailbox_safe.hpp
+3
-8
mutex.hpp
src/mutex.hpp
+11
-1
No files found.
src/mailbox_safe.cpp
View file @
55886b8b
...
@@ -20,8 +20,8 @@
...
@@ -20,8 +20,8 @@
#include "mailbox_safe.hpp"
#include "mailbox_safe.hpp"
#include "err.hpp"
#include "err.hpp"
zmq
::
mailbox_safe_t
::
mailbox_safe_t
(
mutex_t
*
s
ocket_mutex
_
)
:
zmq
::
mailbox_safe_t
::
mailbox_safe_t
(
mutex_t
*
s
ync
_
)
:
s
ocket_mutex
(
socket_mutex
_
)
s
ync
(
sync
_
)
{
{
// Get the pipe into passive state. That way, if the users starts by
// Get the pipe into passive state. That way, if the users starts by
// polling on the associated file descriptor it will get woken up when
// polling on the associated file descriptor it will get woken up when
...
@@ -36,20 +36,17 @@ zmq::mailbox_safe_t::~mailbox_safe_t ()
...
@@ -36,20 +36,17 @@ zmq::mailbox_safe_t::~mailbox_safe_t ()
// Work around problem that other threads might still be in our
// Work around problem that other threads might still be in our
// send() method, by waiting on the mutex before disappearing.
// send() method, by waiting on the mutex before disappearing.
sync
.
lock
();
sync
->
lock
();
sync
.
unlock
();
sync
->
unlock
();
}
}
void
zmq
::
mailbox_safe_t
::
add_signaler
(
signaler_t
*
signaler
)
void
zmq
::
mailbox_safe_t
::
add_signaler
(
signaler_t
*
signaler
)
{
{
sync
.
lock
();
signalers
.
push_back
(
signaler
);
signalers
.
push_back
(
signaler
);
sync
.
unlock
();
}
}
void
zmq
::
mailbox_safe_t
::
remove_signaler
(
signaler_t
*
signaler
)
void
zmq
::
mailbox_safe_t
::
remove_signaler
(
signaler_t
*
signaler
)
{
{
sync
.
lock
();
std
::
vector
<
signaler_t
*>::
iterator
it
=
signalers
.
begin
();
std
::
vector
<
signaler_t
*>::
iterator
it
=
signalers
.
begin
();
// TODO: make a copy of array and signal outside the lock
// TODO: make a copy of array and signal outside the lock
...
@@ -60,25 +57,22 @@ void zmq::mailbox_safe_t::remove_signaler(signaler_t* signaler)
...
@@ -60,25 +57,22 @@ void zmq::mailbox_safe_t::remove_signaler(signaler_t* signaler)
if
(
it
!=
signalers
.
end
())
if
(
it
!=
signalers
.
end
())
signalers
.
erase
(
it
);
signalers
.
erase
(
it
);
sync
.
unlock
();
}
}
void
zmq
::
mailbox_safe_t
::
send
(
const
command_t
&
cmd_
)
void
zmq
::
mailbox_safe_t
::
send
(
const
command_t
&
cmd_
)
{
{
sync
.
lock
();
sync
->
lock
();
cpipe
.
write
(
cmd_
,
false
);
cpipe
.
write
(
cmd_
,
false
);
const
bool
ok
=
cpipe
.
flush
();
const
bool
ok
=
cpipe
.
flush
();
if
(
!
ok
)
{
if
(
!
ok
)
{
cond_var
.
broadcast
();
for
(
std
::
vector
<
signaler_t
*>::
iterator
it
=
signalers
.
begin
();
it
!=
signalers
.
end
();
++
it
){
for
(
std
::
vector
<
signaler_t
*>::
iterator
it
=
signalers
.
begin
();
it
!=
signalers
.
end
();
++
it
){
(
*
it
)
->
send
();
(
*
it
)
->
send
();
}
}
}
}
sync
.
unlock
();
sync
->
unlock
();
if
(
!
ok
)
cond_var
.
broadcast
();
}
}
int
zmq
::
mailbox_safe_t
::
recv
(
command_t
*
cmd_
,
int
timeout_
)
int
zmq
::
mailbox_safe_t
::
recv
(
command_t
*
cmd_
,
int
timeout_
)
...
@@ -86,14 +80,9 @@ int zmq::mailbox_safe_t::recv (command_t *cmd_, int timeout_)
...
@@ -86,14 +80,9 @@ int zmq::mailbox_safe_t::recv (command_t *cmd_, int timeout_)
// Try to get the command straight away.
// Try to get the command straight away.
if
(
cpipe
.
read
(
cmd_
))
if
(
cpipe
.
read
(
cmd_
))
return
0
;
return
0
;
if
(
timeout_
==
0
)
{
errno
=
EAGAIN
;
return
-
1
;
}
// Wait for signal from the command sender.
// Wait for signal from the command sender.
int
rc
=
cond_var
.
wait
(
s
ocket_mutex
,
timeout_
);
int
rc
=
cond_var
.
wait
(
s
ync
,
timeout_
);
if
(
rc
==
-
1
)
{
if
(
rc
==
-
1
)
{
errno_assert
(
errno
==
EAGAIN
||
errno
==
EINTR
);
errno_assert
(
errno
==
EAGAIN
||
errno
==
EINTR
);
return
-
1
;
return
-
1
;
...
...
src/mailbox_safe.hpp
View file @
55886b8b
...
@@ -43,7 +43,7 @@ namespace zmq
...
@@ -43,7 +43,7 @@ namespace zmq
{
{
public
:
public
:
mailbox_safe_t
(
mutex_t
*
s
ocket_mutex
_
);
mailbox_safe_t
(
mutex_t
*
s
ync
_
);
~
mailbox_safe_t
();
~
mailbox_safe_t
();
void
send
(
const
command_t
&
cmd_
);
void
send
(
const
command_t
&
cmd_
);
...
@@ -72,13 +72,8 @@ namespace zmq
...
@@ -72,13 +72,8 @@ namespace zmq
// Condition variable to pass signals from writer thread to reader thread.
// Condition variable to pass signals from writer thread to reader thread.
condition_variable_t
cond_var
;
condition_variable_t
cond_var
;
// There's only one thread receiving from the mailbox, but there
// Synchronize access to the mailbox from receivers and senders
// is arbitrary number of threads sending. Given that ypipe requires
mutex_t
*
sync
;
// synchronised access on both of its endpoints, we have to synchronise
// the sending side.
mutex_t
sync
;
mutex_t
*
socket_mutex
;
std
::
vector
<
zmq
::
signaler_t
*
>
signalers
;
std
::
vector
<
zmq
::
signaler_t
*
>
signalers
;
...
...
src/mutex.hpp
View file @
55886b8b
...
@@ -88,7 +88,13 @@ namespace zmq
...
@@ -88,7 +88,13 @@ namespace zmq
public
:
public
:
inline
mutex_t
()
inline
mutex_t
()
{
{
int
rc
=
pthread_mutex_init
(
&
mutex
,
NULL
);
int
rc
=
pthread_mutexattr_init
(
&
attr
);
posix_assert
(
rc
);
rc
=
pthread_mutexattr_settype
(
&
attr
,
PTHREAD_MUTEX_RECURSIVE
);
posix_assert
(
rc
);
rc
=
pthread_mutex_init
(
&
mutex
,
&
attr
);
posix_assert
(
rc
);
posix_assert
(
rc
);
}
}
...
@@ -96,6 +102,9 @@ namespace zmq
...
@@ -96,6 +102,9 @@ namespace zmq
{
{
int
rc
=
pthread_mutex_destroy
(
&
mutex
);
int
rc
=
pthread_mutex_destroy
(
&
mutex
);
posix_assert
(
rc
);
posix_assert
(
rc
);
rc
=
pthread_mutexattr_destroy
(
&
attr
);
posix_assert
(
rc
);
}
}
inline
void
lock
()
inline
void
lock
()
...
@@ -128,6 +137,7 @@ namespace zmq
...
@@ -128,6 +137,7 @@ namespace zmq
private
:
private
:
pthread_mutex_t
mutex
;
pthread_mutex_t
mutex
;
pthread_mutexattr_t
attr
;
// Disable copy construction and assignment.
// Disable copy construction and assignment.
mutex_t
(
const
mutex_t
&
);
mutex_t
(
const
mutex_t
&
);
...
...
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