summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorIgor Ryzhov <iryzhov@nfware.com>2021-10-14 01:45:14 +0300
committerGitHub <noreply@github.com>2021-10-14 01:45:14 +0300
commitbdee439b626fb23bfe43f9c06a4ef3c7331c9df2 (patch)
tree743a95849536f2a8cb84fb97d7cb8f868252929c
parent645f7d11507ebbe035f9b1e98a41250b554888f6 (diff)
parent6b428a3db6fa72f84066de6b5f47ed7d4bb4242c (diff)
Merge pull request #9767 from FRRouting/mergify/bp/dev/8.1/pr-9345
lib: Fix zmq wrapper module mem handling (backport #9345)
-rw-r--r--lib/frr_zmq.c43
-rw-r--r--lib/frr_zmq.h3
2 files changed, 27 insertions, 19 deletions
diff --git a/lib/frr_zmq.c b/lib/frr_zmq.c
index ea9c828f7c..4e947a8a84 100644
--- a/lib/frr_zmq.c
+++ b/lib/frr_zmq.c
@@ -84,7 +84,10 @@ static int frrzmq_read_msg(struct thread *t)
break;
if (cb->read.cb_msg) {
+ cb->in_cb = true;
cb->read.cb_msg(cb->read.arg, cb->zmqsock);
+ cb->in_cb = false;
+
read = 1;
if (cb->read.cancelled) {
@@ -92,7 +95,8 @@ static int frrzmq_read_msg(struct thread *t)
ZMQ_POLLOUT);
cb->read.thread = NULL;
if (cb->write.cancelled && !cb->write.thread)
- XFREE(MTYPE_ZEROMQ_CB, cb);
+ XFREE(MTYPE_ZEROMQ_CB, *cbp);
+
return 0;
}
continue;
@@ -112,15 +116,19 @@ static int frrzmq_read_msg(struct thread *t)
}
read = 1;
+ cb->in_cb = true;
cb->read.cb_part(cb->read.arg, cb->zmqsock, &msg,
partno);
+ cb->in_cb = false;
+
if (cb->read.cancelled) {
zmq_msg_close(&msg);
frrzmq_check_events(cbp, &cb->write,
ZMQ_POLLOUT);
cb->read.thread = NULL;
if (cb->write.cancelled && !cb->write.thread)
- XFREE(MTYPE_ZEROMQ_CB, cb);
+ XFREE(MTYPE_ZEROMQ_CB, *cbp);
+
return 0;
}
@@ -183,7 +191,6 @@ int _frrzmq_thread_add_read(const struct xref_threadsched *xref,
cb = *cbp;
else {
cb = XCALLOC(MTYPE_ZEROMQ_CB, sizeof(struct frrzmq_cb));
-
cb->write.cancelled = true;
*cbp = cb;
}
@@ -195,6 +202,7 @@ int _frrzmq_thread_add_read(const struct xref_threadsched *xref,
cb->read.cb_part = partfunc;
cb->read.cb_error = errfunc;
cb->read.cancelled = false;
+ cb->in_cb = false;
if (events & ZMQ_POLLIN) {
thread_cancel(&cb->read.thread);
@@ -232,14 +240,18 @@ static int frrzmq_write_msg(struct thread *t)
break;
if (cb->write.cb_msg) {
+ cb->in_cb = true;
cb->write.cb_msg(cb->write.arg, cb->zmqsock);
+ cb->in_cb = false;
+
written = 1;
if (cb->write.cancelled) {
frrzmq_check_events(cbp, &cb->read, ZMQ_POLLIN);
cb->write.thread = NULL;
if (cb->read.cancelled && !cb->read.thread)
- XFREE(MTYPE_ZEROMQ_CB, cb);
+ XFREE(MTYPE_ZEROMQ_CB, *cbp);
+
return 0;
}
continue;
@@ -286,7 +298,6 @@ int _frrzmq_thread_add_write(const struct xref_threadsched *xref,
cb = *cbp;
else {
cb = XCALLOC(MTYPE_ZEROMQ_CB, sizeof(struct frrzmq_cb));
-
cb->read.cancelled = true;
*cbp = cb;
}
@@ -298,6 +309,7 @@ int _frrzmq_thread_add_write(const struct xref_threadsched *xref,
cb->write.cb_part = NULL;
cb->write.cb_error = errfunc;
cb->write.cancelled = false;
+ cb->in_cb = false;
if (events & ZMQ_POLLOUT) {
thread_cancel(&cb->write.thread);
@@ -317,22 +329,15 @@ void frrzmq_thread_cancel(struct frrzmq_cb **cb, struct cb_core *core)
core->cancelled = true;
thread_cancel(&core->thread);
- /*
- * Looking at this code one would assume that FRR
- * would want a `!(*cb)->write.thread. This was
- * attempted in e08165def1c62beee0e87385 but this
- * change caused `make check` to stop working
- * which was not noticed because our CI system
- * does not build with zeromq. Put this back
- * to the code as written in 2017. e08165de..
- * was introduced in 2021. So someone was ok
- * with frrzmq_thread_cancel for 4 years. This will
- * allow those people doing `make check` to continue
- * working. In the meantime if the people using
- * this code see an issue they can fix it
+ /* If cancelled from within a callback, don't try to free memory
+ * in this path.
*/
+ if ((*cb)->in_cb)
+ return;
+
+ /* Ok to free the callback context if no more ... context. */
if ((*cb)->read.cancelled && !(*cb)->read.thread
- && (*cb)->write.cancelled && (*cb)->write.thread)
+ && (*cb)->write.cancelled && ((*cb)->write.thread == NULL))
XFREE(MTYPE_ZEROMQ_CB, *cb);
}
diff --git a/lib/frr_zmq.h b/lib/frr_zmq.h
index d30cf8a841..b3be78cbea 100644
--- a/lib/frr_zmq.h
+++ b/lib/frr_zmq.h
@@ -49,10 +49,13 @@ struct cb_core {
unsigned partnum);
void (*cb_error)(void *arg, void *zmqsock);
};
+
struct frrzmq_cb {
void *zmqsock;
int fd;
+ bool in_cb; /* This context is in a read or write callback. */
+
struct cb_core read;
struct cb_core write;
};