events: Avoid double free possibility on remote call failure

If a remote call fails during event registration (more than likely from
a network failure or remote libvirtd restart timed just right), then when
calling the virObjectEventStateDeregisterID we don't want to call the
registered @freecb function because that breaks our contract that we
would only call it after succesfully returning.  If the @freecb routine
were called, it could result in a double free from properly coded
applications that free their opaque data on failure to register, as seen
in the following details:

    Program terminated with signal 6, Aborted.
    #0  0x00007fc45cba15d7 in raise
    #1  0x00007fc45cba2cc8 in abort
    #2  0x00007fc45cbe12f7 in __libc_message
    #3  0x00007fc45cbe86d3 in _int_free
    #4  0x00007fc45d8d292c in PyDict_Fini
    #5  0x00007fc45d94f46a in Py_Finalize
    #6  0x00007fc45d960735 in Py_Main
    #7  0x00007fc45cb8daf5 in __libc_start_main
    #8  0x0000000000400721 in _start

The double dereference of 'pyobj_cbData' is triggered in the following way:

    (1) libvirt_virConnectDomainEventRegisterAny is invoked.
    (2) the event is successfully added to the event callback list
        (virDomainEventStateRegisterClient in
        remoteConnectDomainEventRegisterAny returns 1 which means ok).
    (3) when function remoteConnectDomainEventRegisterAny is hit,
        network connection disconnected coincidently (or libvirtd is
        restarted) in the context of function 'call' then the connection
        is lost and the function 'call' failed, the branch
        virObjectEventStateDeregisterID is therefore taken.
    (4) 'pyobj_conn' is dereferenced the 1st time in
        libvirt_virConnectDomainEventFreeFunc.
    (5) 'pyobj_cbData' (refered to pyobj_conn) is dereferenced the
         2nd time in libvirt_virConnectDomainEventRegisterAny.
    (6) the double free error is triggered.

Resolve this by adding a @doFreeCb boolean in order to avoid calling the
freeCb in virObjectEventStateDeregisterID for any remote call failure in
a remoteConnect*EventRegister* API. For remoteConnect*EventDeregister* calls,
the passed value would be true indicating they should run the freecb if it
exists; whereas, it's false for the remote call failure path.

Patch based on the investigation and initial patch posted by
fangying <fangying1@huawei.com>.
This commit is contained in:
John Ferlan 2017-06-14 07:32:15 -04:00
parent 70c9b44270
commit 2065499b60
16 changed files with 52 additions and 40 deletions

View File

@ -1503,7 +1503,7 @@ bhyveConnectDomainEventDeregisterAny(virConnectPtr conn,
if (virObjectEventStateDeregisterID(conn, if (virObjectEventStateDeregisterID(conn,
privconn->domainEventState, privconn->domainEventState,
callbackID) < 0) callbackID, true) < 0)
return -1; return -1;
return 0; return 0;

View File

@ -2301,7 +2301,7 @@ virDomainEventStateDeregister(virConnectPtr conn,
NULL); NULL);
if (callbackID < 0) if (callbackID < 0)
return -1; return -1;
return virObjectEventStateDeregisterID(conn, state, callbackID); return virObjectEventStateDeregisterID(conn, state, callbackID, true);
} }

View File

@ -234,13 +234,15 @@ virObjectEventCallbackListCount(virConnectPtr conn,
* @conn: pointer to the connection * @conn: pointer to the connection
* @cbList: the list * @cbList: the list
* @callback: the callback to remove * @callback: the callback to remove
* @doFreeCb: Inhibit calling the freecb
* *
* Internal function to remove a callback from a virObjectEventCallbackListPtr * Internal function to remove a callback from a virObjectEventCallbackListPtr
*/ */
static int static int
virObjectEventCallbackListRemoveID(virConnectPtr conn, virObjectEventCallbackListRemoveID(virConnectPtr conn,
virObjectEventCallbackListPtr cbList, virObjectEventCallbackListPtr cbList,
int callbackID) int callbackID,
bool doFreeCb)
{ {
size_t i; size_t i;
@ -256,7 +258,10 @@ virObjectEventCallbackListRemoveID(virConnectPtr conn,
cb->key_filter ? cb->key : NULL, cb->key_filter ? cb->key : NULL,
cb->remoteID >= 0) - 1); cb->remoteID >= 0) - 1);
if (cb->freecb) /* @doFreeCb inhibits calling @freecb from error paths in
* register functions to ensure the caller of a failed register
* function won't end up with a double free error */
if (doFreeCb && cb->freecb)
(*cb->freecb)(cb->opaque); (*cb->freecb)(cb->opaque);
virObjectEventCallbackFree(cb); virObjectEventCallbackFree(cb);
VIR_DELETE_ELEMENT(cbList->callbacks, i, cbList->count); VIR_DELETE_ELEMENT(cbList->callbacks, i, cbList->count);
@ -927,16 +932,22 @@ virObjectEventStateRegisterID(virConnectPtr conn,
* @conn: connection to associate with callback * @conn: connection to associate with callback
* @state: object event state * @state: object event state
* @callbackID: ID of the function to remove from event * @callbackID: ID of the function to remove from event
* @doFreeCb: Allow the calling of a freecb
* *
* Unregister the function @callbackID with connection @conn, * Unregister the function @callbackID with connection @conn,
* from @state, for events. * from @state, for events. If @doFreeCb is false, then we
* are being called from a remote call failure path for the
* Event registration indicating a -1 return to the caller. The
* caller wouldn't expect us to run their freecb function if it
* exists, so we cannot do so.
* *
* Returns: the number of callbacks still registered, or -1 on error * Returns: the number of callbacks still registered, or -1 on error
*/ */
int int
virObjectEventStateDeregisterID(virConnectPtr conn, virObjectEventStateDeregisterID(virConnectPtr conn,
virObjectEventStatePtr state, virObjectEventStatePtr state,
int callbackID) int callbackID,
bool doFreeCb)
{ {
int ret; int ret;
@ -946,8 +957,8 @@ virObjectEventStateDeregisterID(virConnectPtr conn,
state->callbacks, state->callbacks,
callbackID); callbackID);
else else
ret = virObjectEventCallbackListRemoveID(conn, ret = virObjectEventCallbackListRemoveID(conn, state->callbacks,
state->callbacks, callbackID); callbackID, doFreeCb);
virObjectEventStateCleanupTimer(state, true); virObjectEventStateCleanupTimer(state, true);

View File

@ -73,7 +73,8 @@ virObjectEventStateQueueRemote(virObjectEventStatePtr state,
int int
virObjectEventStateDeregisterID(virConnectPtr conn, virObjectEventStateDeregisterID(virConnectPtr conn,
virObjectEventStatePtr state, virObjectEventStatePtr state,
int callbackID) int callbackID,
bool doFreeCb)
ATTRIBUTE_NONNULL(1) ATTRIBUTE_NONNULL(2); ATTRIBUTE_NONNULL(1) ATTRIBUTE_NONNULL(2);
int int

View File

@ -5652,7 +5652,7 @@ libxlConnectDomainEventDeregisterAny(virConnectPtr conn, int callbackID)
if (virObjectEventStateDeregisterID(conn, if (virObjectEventStateDeregisterID(conn,
driver->domainEventState, driver->domainEventState,
callbackID) < 0) callbackID, true) < 0)
return -1; return -1;
return 0; return 0;

View File

@ -1479,7 +1479,7 @@ lxcConnectDomainEventDeregisterAny(virConnectPtr conn,
if (virObjectEventStateDeregisterID(conn, if (virObjectEventStateDeregisterID(conn,
driver->domainEventState, driver->domainEventState,
callbackID) < 0) callbackID, true) < 0)
return -1; return -1;
return 0; return 0;

View File

@ -3001,7 +3001,7 @@ networkConnectNetworkEventDeregisterAny(virConnectPtr conn,
if (virObjectEventStateDeregisterID(conn, if (virObjectEventStateDeregisterID(conn,
driver->networkEventState, driver->networkEventState,
callbackID) < 0) callbackID, true) < 0)
goto cleanup; goto cleanup;
ret = 0; ret = 0;

View File

@ -690,7 +690,7 @@ nodeConnectNodeDeviceEventDeregisterAny(virConnectPtr conn,
if (virObjectEventStateDeregisterID(conn, if (virObjectEventStateDeregisterID(conn,
driver->nodeDeviceEventState, driver->nodeDeviceEventState,
callbackID) < 0) callbackID, true) < 0)
goto cleanup; goto cleanup;
ret = 0; ret = 0;

View File

@ -11826,7 +11826,7 @@ qemuConnectDomainEventDeregisterAny(virConnectPtr conn,
if (virObjectEventStateDeregisterID(conn, if (virObjectEventStateDeregisterID(conn,
driver->domainEventState, driver->domainEventState,
callbackID) < 0) callbackID, true) < 0)
goto cleanup; goto cleanup;
ret = 0; ret = 0;
@ -18472,7 +18472,7 @@ qemuConnectDomainQemuMonitorEventDeregister(virConnectPtr conn,
goto cleanup; goto cleanup;
if (virObjectEventStateDeregisterID(conn, driver->domainEventState, if (virObjectEventStateDeregisterID(conn, driver->domainEventState,
callbackID) < 0) callbackID, true) < 0)
goto cleanup; goto cleanup;
ret = 0; ret = 0;

View File

@ -3066,7 +3066,7 @@ remoteConnectNetworkEventRegisterAny(virConnectPtr conn,
(xdrproc_t) xdr_remote_connect_network_event_register_any_args, (char *) &args, (xdrproc_t) xdr_remote_connect_network_event_register_any_args, (char *) &args,
(xdrproc_t) xdr_remote_connect_network_event_register_any_ret, (char *) &ret) == -1) { (xdrproc_t) xdr_remote_connect_network_event_register_any_ret, (char *) &ret) == -1) {
virObjectEventStateDeregisterID(conn, priv->eventState, virObjectEventStateDeregisterID(conn, priv->eventState,
callbackID); callbackID, false);
goto done; goto done;
} }
virObjectEventStateSetRemote(conn, priv->eventState, callbackID, virObjectEventStateSetRemote(conn, priv->eventState, callbackID,
@ -3099,7 +3099,7 @@ remoteConnectNetworkEventDeregisterAny(virConnectPtr conn,
goto done; goto done;
if ((count = virObjectEventStateDeregisterID(conn, priv->eventState, if ((count = virObjectEventStateDeregisterID(conn, priv->eventState,
callbackID)) < 0) callbackID, true)) < 0)
goto done; goto done;
/* If that was the last callback for this eventID, we need to disable /* If that was the last callback for this eventID, we need to disable
@ -3160,7 +3160,7 @@ remoteConnectStoragePoolEventRegisterAny(virConnectPtr conn,
(xdrproc_t) xdr_remote_connect_storage_pool_event_register_any_args, (char *) &args, (xdrproc_t) xdr_remote_connect_storage_pool_event_register_any_args, (char *) &args,
(xdrproc_t) xdr_remote_connect_storage_pool_event_register_any_ret, (char *) &ret) == -1) { (xdrproc_t) xdr_remote_connect_storage_pool_event_register_any_ret, (char *) &ret) == -1) {
virObjectEventStateDeregisterID(conn, priv->eventState, virObjectEventStateDeregisterID(conn, priv->eventState,
callbackID); callbackID, false);
goto done; goto done;
} }
@ -3193,7 +3193,7 @@ remoteConnectStoragePoolEventDeregisterAny(virConnectPtr conn,
goto done; goto done;
if ((count = virObjectEventStateDeregisterID(conn, priv->eventState, if ((count = virObjectEventStateDeregisterID(conn, priv->eventState,
callbackID)) < 0) callbackID, true)) < 0)
goto done; goto done;
/* If that was the last callback for this eventID, we need to disable /* If that was the last callback for this eventID, we need to disable
@ -3256,7 +3256,7 @@ remoteConnectNodeDeviceEventRegisterAny(virConnectPtr conn,
(xdrproc_t) xdr_remote_connect_node_device_event_register_any_args, (char *) &args, (xdrproc_t) xdr_remote_connect_node_device_event_register_any_args, (char *) &args,
(xdrproc_t) xdr_remote_connect_node_device_event_register_any_ret, (char *) &ret) == -1) { (xdrproc_t) xdr_remote_connect_node_device_event_register_any_ret, (char *) &ret) == -1) {
virObjectEventStateDeregisterID(conn, priv->eventState, virObjectEventStateDeregisterID(conn, priv->eventState,
callbackID); callbackID, false);
goto done; goto done;
} }
@ -3290,7 +3290,7 @@ remoteConnectNodeDeviceEventDeregisterAny(virConnectPtr conn,
goto done; goto done;
if ((count = virObjectEventStateDeregisterID(conn, priv->eventState, if ((count = virObjectEventStateDeregisterID(conn, priv->eventState,
callbackID)) < 0) callbackID, true)) < 0)
goto done; goto done;
/* If that was the last callback for this eventID, we need to disable /* If that was the last callback for this eventID, we need to disable
@ -3353,7 +3353,7 @@ remoteConnectSecretEventRegisterAny(virConnectPtr conn,
(xdrproc_t) xdr_remote_connect_secret_event_register_any_args, (char *) &args, (xdrproc_t) xdr_remote_connect_secret_event_register_any_args, (char *) &args,
(xdrproc_t) xdr_remote_connect_secret_event_register_any_ret, (char *) &ret) == -1) { (xdrproc_t) xdr_remote_connect_secret_event_register_any_ret, (char *) &ret) == -1) {
virObjectEventStateDeregisterID(conn, priv->eventState, virObjectEventStateDeregisterID(conn, priv->eventState,
callbackID); callbackID, false);
goto done; goto done;
} }
@ -3387,7 +3387,7 @@ remoteConnectSecretEventDeregisterAny(virConnectPtr conn,
goto done; goto done;
if ((count = virObjectEventStateDeregisterID(conn, priv->eventState, if ((count = virObjectEventStateDeregisterID(conn, priv->eventState,
callbackID)) < 0) callbackID, true)) < 0)
goto done; goto done;
/* If that was the last callback for this eventID, we need to disable /* If that was the last callback for this eventID, we need to disable
@ -3453,7 +3453,7 @@ remoteConnectDomainQemuMonitorEventRegister(virConnectPtr conn,
(xdrproc_t) xdr_qemu_connect_domain_monitor_event_register_args, (char *) &args, (xdrproc_t) xdr_qemu_connect_domain_monitor_event_register_args, (char *) &args,
(xdrproc_t) xdr_qemu_connect_domain_monitor_event_register_ret, (char *) &ret) == -1) { (xdrproc_t) xdr_qemu_connect_domain_monitor_event_register_ret, (char *) &ret) == -1) {
virObjectEventStateDeregisterID(conn, priv->eventState, virObjectEventStateDeregisterID(conn, priv->eventState,
callbackID); callbackID, false);
goto done; goto done;
} }
virObjectEventStateSetRemote(conn, priv->eventState, callbackID, virObjectEventStateSetRemote(conn, priv->eventState, callbackID,
@ -3485,7 +3485,7 @@ remoteConnectDomainQemuMonitorEventDeregister(virConnectPtr conn,
goto done; goto done;
if ((count = virObjectEventStateDeregisterID(conn, priv->eventState, if ((count = virObjectEventStateDeregisterID(conn, priv->eventState,
callbackID)) < 0) callbackID, true)) < 0)
goto done; goto done;
/* If that was the last callback for this event, we need to disable /* If that was the last callback for this event, we need to disable
@ -4409,7 +4409,7 @@ remoteConnectDomainEventRegister(virConnectPtr conn,
(xdrproc_t) xdr_remote_connect_domain_event_callback_register_any_args, (char *) &args, (xdrproc_t) xdr_remote_connect_domain_event_callback_register_any_args, (char *) &args,
(xdrproc_t) xdr_remote_connect_domain_event_callback_register_any_ret, (char *) &ret) == -1) { (xdrproc_t) xdr_remote_connect_domain_event_callback_register_any_ret, (char *) &ret) == -1) {
virObjectEventStateDeregisterID(conn, priv->eventState, virObjectEventStateDeregisterID(conn, priv->eventState,
callbackID); callbackID, false);
goto done; goto done;
} }
virObjectEventStateSetRemote(conn, priv->eventState, callbackID, virObjectEventStateSetRemote(conn, priv->eventState, callbackID,
@ -4419,7 +4419,7 @@ remoteConnectDomainEventRegister(virConnectPtr conn,
(xdrproc_t) xdr_void, (char *) NULL, (xdrproc_t) xdr_void, (char *) NULL,
(xdrproc_t) xdr_void, (char *) NULL) == -1) { (xdrproc_t) xdr_void, (char *) NULL) == -1) {
virObjectEventStateDeregisterID(conn, priv->eventState, virObjectEventStateDeregisterID(conn, priv->eventState,
callbackID); callbackID, false);
goto done; goto done;
} }
} }
@ -4452,7 +4452,7 @@ remoteConnectDomainEventDeregister(virConnectPtr conn,
goto done; goto done;
if ((count = virObjectEventStateDeregisterID(conn, priv->eventState, if ((count = virObjectEventStateDeregisterID(conn, priv->eventState,
callbackID)) < 0) callbackID, true)) < 0)
goto done; goto done;
if (count == 0) { if (count == 0) {
@ -5951,7 +5951,7 @@ remoteConnectDomainEventRegisterAny(virConnectPtr conn,
(xdrproc_t) xdr_remote_connect_domain_event_callback_register_any_args, (char *) &args, (xdrproc_t) xdr_remote_connect_domain_event_callback_register_any_args, (char *) &args,
(xdrproc_t) xdr_remote_connect_domain_event_callback_register_any_ret, (char *) &ret) == -1) { (xdrproc_t) xdr_remote_connect_domain_event_callback_register_any_ret, (char *) &ret) == -1) {
virObjectEventStateDeregisterID(conn, priv->eventState, virObjectEventStateDeregisterID(conn, priv->eventState,
callbackID); callbackID, false);
goto done; goto done;
} }
virObjectEventStateSetRemote(conn, priv->eventState, callbackID, virObjectEventStateSetRemote(conn, priv->eventState, callbackID,
@ -5965,7 +5965,7 @@ remoteConnectDomainEventRegisterAny(virConnectPtr conn,
(xdrproc_t) xdr_remote_connect_domain_event_register_any_args, (char *) &args, (xdrproc_t) xdr_remote_connect_domain_event_register_any_args, (char *) &args,
(xdrproc_t) xdr_void, (char *)NULL) == -1) { (xdrproc_t) xdr_void, (char *)NULL) == -1) {
virObjectEventStateDeregisterID(conn, priv->eventState, virObjectEventStateDeregisterID(conn, priv->eventState,
callbackID); callbackID, false);
goto done; goto done;
} }
} }
@ -5996,7 +5996,7 @@ remoteConnectDomainEventDeregisterAny(virConnectPtr conn,
goto done; goto done;
if ((count = virObjectEventStateDeregisterID(conn, priv->eventState, if ((count = virObjectEventStateDeregisterID(conn, priv->eventState,
callbackID)) < 0) callbackID, true)) < 0)
goto done; goto done;
/* If that was the last callback for this eventID, we need to disable /* If that was the last callback for this eventID, we need to disable

View File

@ -532,7 +532,7 @@ secretConnectSecretEventDeregisterAny(virConnectPtr conn,
if (virObjectEventStateDeregisterID(conn, if (virObjectEventStateDeregisterID(conn,
driver->secretEventState, driver->secretEventState,
callbackID) < 0) callbackID, true) < 0)
goto cleanup; goto cleanup;
ret = 0; ret = 0;

View File

@ -2662,7 +2662,7 @@ storageConnectStoragePoolEventDeregisterAny(virConnectPtr conn,
if (virObjectEventStateDeregisterID(conn, if (virObjectEventStateDeregisterID(conn,
driver->storageEventState, driver->storageEventState,
callbackID) < 0) callbackID, true) < 0)
goto cleanup; goto cleanup;
ret = 0; ret = 0;

View File

@ -5697,7 +5697,7 @@ testConnectDomainEventDeregisterAny(virConnectPtr conn,
int ret = 0; int ret = 0;
if (virObjectEventStateDeregisterID(conn, driver->eventState, if (virObjectEventStateDeregisterID(conn, driver->eventState,
callbackID) < 0) callbackID, true) < 0)
ret = -1; ret = -1;
return ret; return ret;
@ -5731,7 +5731,7 @@ testConnectNetworkEventDeregisterAny(virConnectPtr conn,
int ret = 0; int ret = 0;
if (virObjectEventStateDeregisterID(conn, driver->eventState, if (virObjectEventStateDeregisterID(conn, driver->eventState,
callbackID) < 0) callbackID, true) < 0)
ret = -1; ret = -1;
return ret; return ret;
@ -5764,7 +5764,7 @@ testConnectStoragePoolEventDeregisterAny(virConnectPtr conn,
int ret = 0; int ret = 0;
if (virObjectEventStateDeregisterID(conn, driver->eventState, if (virObjectEventStateDeregisterID(conn, driver->eventState,
callbackID) < 0) callbackID, true) < 0)
ret = -1; ret = -1;
return ret; return ret;
@ -5797,7 +5797,7 @@ testConnectNodeDeviceEventDeregisterAny(virConnectPtr conn,
int ret = 0; int ret = 0;
if (virObjectEventStateDeregisterID(conn, driver->eventState, if (virObjectEventStateDeregisterID(conn, driver->eventState,
callbackID) < 0) callbackID, true) < 0)
ret = -1; ret = -1;
return ret; return ret;

View File

@ -2721,7 +2721,7 @@ umlConnectDomainEventDeregisterAny(virConnectPtr conn,
umlDriverLock(driver); umlDriverLock(driver);
if (virObjectEventStateDeregisterID(conn, if (virObjectEventStateDeregisterID(conn,
driver->domainEventState, driver->domainEventState,
callbackID) < 0) callbackID, true) < 0)
ret = -1; ret = -1;
umlDriverUnlock(driver); umlDriverUnlock(driver);

View File

@ -1053,7 +1053,7 @@ vzConnectDomainEventDeregisterAny(virConnectPtr conn,
if (virObjectEventStateDeregisterID(conn, if (virObjectEventStateDeregisterID(conn,
privconn->driver->domainEventState, privconn->driver->domainEventState,
callbackID) < 0) callbackID, true) < 0)
return -1; return -1;
return 0; return 0;

View File

@ -2284,7 +2284,7 @@ xenUnifiedConnectDomainEventDeregisterAny(virConnectPtr conn,
if (virObjectEventStateDeregisterID(conn, if (virObjectEventStateDeregisterID(conn,
priv->domainEvents, priv->domainEvents,
callbackID) < 0) callbackID, true) < 0)
ret = -1; ret = -1;
xenUnifiedUnlock(priv); xenUnifiedUnlock(priv);