Commit 2a3fb0aa3c (lacp: Don't lock potentially uninitialized mutex in lacp_status().) fixed one bug related to acquiring the file scope 'mutex' without initializing it. However, there was at least one other, in lacp_unixctl_show(). One could just fix that one problem, but that leaves the possibility that I might have missed one or two more. This commit fixes the problem for good, by adding a helper that initializes the mutex and then acquires it.
It's not entirely clear why 'mutex' is a recursive mutex. I think that it might be just because of the callback in lacp_run(). An alternate fix, therefore, would be to eliminate the callback and therefore the need for runtime initialization of the mutex. Bug #1245659. Reported-by: Jeffrey Merrick <jmerr...@vmware.com> Signed-off-by: Ben Pfaff <b...@nicira.com> --- lib/lacp.c | 76 +++++++++++++++++++++++++++++++++++------------------------- 1 file changed, 44 insertions(+), 32 deletions(-) diff --git a/lib/lacp.c b/lib/lacp.c index 49ae5e5..0d30e51 100644 --- a/lib/lacp.c +++ b/lib/lacp.c @@ -203,25 +203,37 @@ lacp_init(void) lacp_unixctl_show, NULL); } -/* Creates a LACP object. */ -struct lacp * -lacp_create(void) OVS_EXCLUDED(mutex) +static void +lacp_lock(void) OVS_ACQUIRES(mutex) { static struct ovsthread_once once = OVSTHREAD_ONCE_INITIALIZER; - struct lacp *lacp; if (ovsthread_once_start(&once)) { ovs_mutex_init_recursive(&mutex); ovsthread_once_done(&once); } + ovs_mutex_lock(&mutex); +} + +static void +lacp_unlock(void) OVS_RELEASES(mutex) +{ + ovs_mutex_unlock(&mutex); +} + +/* Creates a LACP object. */ +struct lacp * +lacp_create(void) OVS_EXCLUDED(mutex) +{ + struct lacp *lacp; lacp = xzalloc(sizeof *lacp); hmap_init(&lacp->slaves); ovs_refcount_init(&lacp->ref_cnt); - ovs_mutex_lock(&mutex); + lacp_lock(); list_push_back(all_lacps, &lacp->node); - ovs_mutex_unlock(&mutex); + lacp_unlock(); return lacp; } @@ -242,7 +254,7 @@ lacp_unref(struct lacp *lacp) OVS_EXCLUDED(mutex) if (lacp && ovs_refcount_unref(&lacp->ref_cnt) == 1) { struct slave *slave, *next; - ovs_mutex_lock(&mutex); + lacp_lock(); HMAP_FOR_EACH_SAFE (slave, next, node, &lacp->slaves) { slave_destroy(slave); } @@ -251,7 +263,7 @@ lacp_unref(struct lacp *lacp) OVS_EXCLUDED(mutex) list_remove(&lacp->node); free(lacp->name); free(lacp); - ovs_mutex_unlock(&mutex); + lacp_unlock(); } } @@ -262,7 +274,7 @@ lacp_configure(struct lacp *lacp, const struct lacp_settings *s) { ovs_assert(!eth_addr_is_zero(s->id)); - ovs_mutex_lock(&mutex); + lacp_lock(); if (!lacp->name || strcmp(s->name, lacp->name)) { free(lacp->name); lacp->name = xstrdup(s->name); @@ -283,7 +295,7 @@ lacp_configure(struct lacp *lacp, const struct lacp_settings *s) lacp->update = true; } - ovs_mutex_unlock(&mutex); + lacp_unlock(); } /* Returns true if 'lacp' is configured in active mode, false if 'lacp' is @@ -292,9 +304,9 @@ bool lacp_is_active(const struct lacp *lacp) OVS_EXCLUDED(mutex) { bool ret; - ovs_mutex_lock(&mutex); + lacp_lock(); ret = lacp->active; - ovs_mutex_unlock(&mutex); + lacp_unlock(); return ret; } @@ -311,7 +323,7 @@ lacp_process_packet(struct lacp *lacp, const void *slave_, long long int tx_rate; struct slave *slave; - ovs_mutex_lock(&mutex); + lacp_lock(); slave = slave_lookup(lacp, slave_); if (!slave) { goto out; @@ -338,7 +350,7 @@ lacp_process_packet(struct lacp *lacp, const void *slave_, } out: - ovs_mutex_unlock(&mutex); + lacp_unlock(); } /* Returns the lacp_status of the given 'lacp' object (which may be NULL). */ @@ -348,9 +360,9 @@ lacp_status(const struct lacp *lacp) OVS_EXCLUDED(mutex) if (lacp) { enum lacp_status ret; - ovs_mutex_lock(&mutex); + lacp_lock(); ret = lacp->negotiated ? LACP_NEGOTIATED : LACP_CONFIGURED; - ovs_mutex_unlock(&mutex); + lacp_unlock(); return ret; } else { /* Don't take 'mutex'. It might not even be initialized, since we @@ -369,7 +381,7 @@ lacp_slave_register(struct lacp *lacp, void *slave_, { struct slave *slave; - ovs_mutex_lock(&mutex); + lacp_lock(); slave = slave_lookup(lacp, slave_); if (!slave) { slave = xzalloc(sizeof *slave); @@ -401,7 +413,7 @@ lacp_slave_register(struct lacp *lacp, void *slave_, slave_set_expired(slave); } } - ovs_mutex_unlock(&mutex); + lacp_unlock(); } /* Unregisters 'slave_' with 'lacp'. */ @@ -411,13 +423,13 @@ lacp_slave_unregister(struct lacp *lacp, const void *slave_) { struct slave *slave; - ovs_mutex_lock(&mutex); + lacp_lock(); slave = slave_lookup(lacp, slave_); if (slave) { slave_destroy(slave); lacp->update = true; } - ovs_mutex_unlock(&mutex); + lacp_unlock(); } /* This function should be called whenever the carrier status of 'slave_' has @@ -431,7 +443,7 @@ lacp_slave_carrier_changed(const struct lacp *lacp, const void *slave_) return; } - ovs_mutex_lock(&mutex); + lacp_lock(); slave = slave_lookup(lacp, slave_); if (!slave) { goto out; @@ -442,7 +454,7 @@ lacp_slave_carrier_changed(const struct lacp *lacp, const void *slave_) } out: - ovs_mutex_unlock(&mutex); + lacp_unlock(); } static bool @@ -466,10 +478,10 @@ lacp_slave_may_enable(const struct lacp *lacp, const void *slave_) struct slave *slave; bool ret; - ovs_mutex_lock(&mutex); + lacp_lock(); slave = slave_lookup(lacp, slave_); ret = slave ? slave_may_enable__(slave) : false; - ovs_mutex_unlock(&mutex); + lacp_unlock(); return ret; } else { return true; @@ -486,10 +498,10 @@ lacp_slave_is_current(const struct lacp *lacp, const void *slave_) struct slave *slave; bool ret; - ovs_mutex_lock(&mutex); + lacp_lock(); slave = slave_lookup(lacp, slave_); ret = slave ? slave->status != LACP_DEFAULTED : false; - ovs_mutex_unlock(&mutex); + lacp_unlock(); return ret; } @@ -499,7 +511,7 @@ lacp_run(struct lacp *lacp, lacp_send_pdu *send_pdu) OVS_EXCLUDED(mutex) { struct slave *slave; - ovs_mutex_lock(&mutex); + lacp_lock(); HMAP_FOR_EACH (slave, node, &lacp->slaves) { if (timer_expired(&slave->rx)) { enum slave_status old_status = slave->status; @@ -545,7 +557,7 @@ lacp_run(struct lacp *lacp, lacp_send_pdu *send_pdu) OVS_EXCLUDED(mutex) seq_change(connectivity_seq_get()); } } - ovs_mutex_unlock(&mutex); + lacp_unlock(); } /* Causes poll_block() to wake up when lacp_run() needs to be called again. */ @@ -554,7 +566,7 @@ lacp_wait(struct lacp *lacp) OVS_EXCLUDED(mutex) { struct slave *slave; - ovs_mutex_lock(&mutex); + lacp_lock(); HMAP_FOR_EACH (slave, node, &lacp->slaves) { if (slave_may_tx(slave)) { timer_wait(&slave->tx); @@ -564,7 +576,7 @@ lacp_wait(struct lacp *lacp) OVS_EXCLUDED(mutex) timer_wait(&slave->rx); } } - ovs_mutex_unlock(&mutex); + lacp_unlock(); } /* Static Helpers. */ @@ -946,7 +958,7 @@ lacp_unixctl_show(struct unixctl_conn *conn, int argc, const char *argv[], struct ds ds = DS_EMPTY_INITIALIZER; struct lacp *lacp; - ovs_mutex_lock(&mutex); + lacp_lock(); if (argc > 1) { lacp = lacp_find(argv[1]); if (!lacp) { @@ -964,5 +976,5 @@ lacp_unixctl_show(struct unixctl_conn *conn, int argc, const char *argv[], ds_destroy(&ds); out: - ovs_mutex_unlock(&mutex); + lacp_unlock(); } -- 1.7.10.4 _______________________________________________ dev mailing list dev@openvswitch.org http://openvswitch.org/mailman/listinfo/dev