提交 054c6d27 编写于 作者: M Michal Privoznik

virnetdaemon: Don't deadlock when talking to D-Bus

https://bugzilla.redhat.com/show_bug.cgi?id=1487322

In ace45e67 I tried to fix a problem that we get the reply to
a D-Bus call while we were sleeping. In that case the callback
was never set. So I changed the code that the callback is called
directly in this case. However, I hadn't realized that since the
callback is called out of order it locks the virNetDaemon.
Exactly the very same virNetDaemon object that we are dealing
with right now and that we have locked already (in
virNetDaemonAddShutdownInhibition())
Signed-off-by: NMichal Privoznik <mprivozn@redhat.com>
上级 9820756c
...@@ -439,14 +439,12 @@ virNetDaemonAutoShutdown(virNetDaemonPtr dmn, ...@@ -439,14 +439,12 @@ virNetDaemonAutoShutdown(virNetDaemonPtr dmn,
#if defined(WITH_DBUS) && defined(DBUS_TYPE_UNIX_FD) #if defined(WITH_DBUS) && defined(DBUS_TYPE_UNIX_FD)
static void static void
virNetDaemonGotInhibitReply(DBusPendingCall *pending, virNetDaemonGotInhibitReplyLocked(DBusPendingCall *pending,
void *opaque) virNetDaemonPtr dmn)
{ {
virNetDaemonPtr dmn = opaque;
DBusMessage *reply; DBusMessage *reply;
int fd; int fd;
virObjectLock(dmn);
dmn->autoShutdownCallingInhibit = false; dmn->autoShutdownCallingInhibit = false;
VIR_DEBUG("dmn=%p", dmn); VIR_DEBUG("dmn=%p", dmn);
...@@ -470,11 +468,22 @@ virNetDaemonGotInhibitReply(DBusPendingCall *pending, ...@@ -470,11 +468,22 @@ virNetDaemonGotInhibitReply(DBusPendingCall *pending,
virDBusMessageUnref(reply); virDBusMessageUnref(reply);
cleanup: cleanup:
virObjectUnlock(dmn);
dbus_pending_call_unref(pending); dbus_pending_call_unref(pending);
} }
static void
virNetDaemonGotInhibitReply(DBusPendingCall *pending,
void *opaque)
{
virNetDaemonPtr dmn = opaque;
virObjectLock(dmn);
virNetDaemonGotInhibitReplyLocked(pending, dmn);
virObjectUnlock(dmn);
}
/* As per: http://www.freedesktop.org/wiki/Software/systemd/inhibit */ /* As per: http://www.freedesktop.org/wiki/Software/systemd/inhibit */
static void static void
virNetDaemonCallInhibit(virNetDaemonPtr dmn, virNetDaemonCallInhibit(virNetDaemonPtr dmn,
...@@ -516,7 +525,7 @@ virNetDaemonCallInhibit(virNetDaemonPtr dmn, ...@@ -516,7 +525,7 @@ virNetDaemonCallInhibit(virNetDaemonPtr dmn,
25 * 1000) && 25 * 1000) &&
pendingReply) { pendingReply) {
if (dbus_pending_call_get_completed(pendingReply)) { if (dbus_pending_call_get_completed(pendingReply)) {
virNetDaemonGotInhibitReply(pendingReply, dmn); virNetDaemonGotInhibitReplyLocked(pendingReply, dmn);
} else { } else {
dbus_pending_call_set_notify(pendingReply, dbus_pending_call_set_notify(pendingReply,
virNetDaemonGotInhibitReply, virNetDaemonGotInhibitReply,
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册