summaryrefslogtreecommitdiff
path: root/cpp/src
diff options
context:
space:
mode:
authorAlan Conway <aconway@apache.org>2008-12-02 21:43:24 +0000
committerAlan Conway <aconway@apache.org>2008-12-02 21:43:24 +0000
commit081d94d64fa41df6f9661ad31afca0ad71fe9d12 (patch)
tree6a4fe30676916a45a8ab2f24a62f768c0e4162c4 /cpp/src
parent7cdb9a9ab688988e596d9fce116a0998decd0972 (diff)
downloadqpid-python-081d94d64fa41df6f9661ad31afca0ad71fe9d12.tar.gz
PollableQueue: fix unsafe use of deque
git-svn-id: https://svn.apache.org/repos/asf/incubator/qpid/trunk/qpid@722622 13f79535-47bb-0310-9956-ffa450edef68
Diffstat (limited to 'cpp/src')
-rw-r--r--cpp/src/qpid/cluster/Cpg.cpp9
-rw-r--r--cpp/src/qpid/cluster/Event.cpp4
-rw-r--r--cpp/src/qpid/sys/PollableQueue.h12
3 files changed, 13 insertions, 12 deletions
diff --git a/cpp/src/qpid/cluster/Cpg.cpp b/cpp/src/qpid/cluster/Cpg.cpp
index 6a9c97139a..0d7b15cfa8 100644
--- a/cpp/src/qpid/cluster/Cpg.cpp
+++ b/cpp/src/qpid/cluster/Cpg.cpp
@@ -103,11 +103,6 @@ bool Cpg::isFlowControlEnabled() {
}
bool Cpg::mcast(const iovec* iov, int iovLen) {
- // Thread-safety note : the cpg_ calls are thread safe, but there
- // is a race below between calling cpg_flow_control_state_get()
- // and calling mcast_joined() where N threads could see the state
- // as disabled and call mcast, but only M < N messages can be sent
- // without exceeding flow control limits.
if (isFlowControlEnabled()) {
QPID_LOG(warning, "CPG flow control enabled")
return false;
@@ -135,13 +130,13 @@ void Cpg::dispatch(cpg_dispatch_t type) {
string Cpg::errorStr(cpg_error_t err, const std::string& msg) {
switch (err) {
case CPG_OK: return msg+": ok";
- case CPG_ERR_LIBRARY: return msg+": library";
+ case CPG_ERR_LIBRARY: return msg+": library error";
case CPG_ERR_TIMEOUT: return msg+": timeout";
case CPG_ERR_TRY_AGAIN: return msg+": timeout. The aisexec daemon may not be running";
case CPG_ERR_INVALID_PARAM: return msg+": invalid param";
case CPG_ERR_NO_MEMORY: return msg+": no memory";
case CPG_ERR_BAD_HANDLE: return msg+": bad handle";
- case CPG_ERR_ACCESS: return msg+": access denied. You may need to set your group ID to 'ais'";
+ case CPG_ERR_ACCESS: return msg+": access denied.";
case CPG_ERR_NOT_EXIST: return msg+": not exist";
case CPG_ERR_EXIST: return msg+": exist";
case CPG_ERR_NOT_SUPPORTED: return msg+": not supported";
diff --git a/cpp/src/qpid/cluster/Event.cpp b/cpp/src/qpid/cluster/Event.cpp
index 87cc7e7bd3..b787ef0f67 100644
--- a/cpp/src/qpid/cluster/Event.cpp
+++ b/cpp/src/qpid/cluster/Event.cpp
@@ -39,7 +39,8 @@ Event::Event(EventType t, const ConnectionId& c, size_t s, uint32_t i)
Event Event::delivered(const MemberId& m, void* d, size_t s) {
Buffer buf(static_cast<char*>(d), s);
- EventType type((EventType)buf.getOctet());
+ EventType type((EventType)buf.getOctet());
+ assert(type == DATA || type == CONTROL);
ConnectionId connection(m, reinterpret_cast<Connection*>(buf.getLongLong()));
uint32_t id = buf.getLong();
assert(buf.getPosition() == OVERHEAD);
@@ -62,6 +63,7 @@ bool Event::mcast (Cpg& cpg) const {
b.putOctet(type);
b.putLongLong(reinterpret_cast<uint64_t>(connectionId.getPointer()));
b.putLong(id);
+ assert(buf.getPosition() == OVERHEAD);
iovec iov[] = { { header, OVERHEAD }, { const_cast<char*>(getData()), getSize() } };
return cpg.mcast(iov, sizeof(iov)/sizeof(*iov));
}
diff --git a/cpp/src/qpid/sys/PollableQueue.h b/cpp/src/qpid/sys/PollableQueue.h
index 2ee29db022..953d198fb0 100644
--- a/cpp/src/qpid/sys/PollableQueue.h
+++ b/cpp/src/qpid/sys/PollableQueue.h
@@ -121,14 +121,18 @@ template <class T> void PollableQueue<T>::dispatch(sys::DispatchHandle& h) {
assert(dispatcher.id() == 0 || dispatcher.id() == Thread::current().id());
dispatcher = Thread::current();
while (!stopped && !queue.empty()) {
+ T value = queue.front();
+ queue.pop_front();
bool ok = false;
{ // unlock to allow concurrent push or call to stop() in callback.
ScopedUnlock u(lock);
- // FIXME aconway 2008-12-02: exception-safe if callback throws.
- ok = callback(queue.front());
+ // FIXME aconway 2008-12-02: not exception safe if callback throws.
+ ok = callback(value);
+ }
+ if (!ok) { // callback cannot process value, put it back.
+ queue.push_front(value);
+ stopped=true;
}
- if (ok) queue.pop_front();
- else stopped=true;
}
dispatcher = Thread();
if (queue.empty()) condition.clear();