From 67195401c915b23e092f98c89ab5f5580a9cf3f6 Mon Sep 17 00:00:00 2001 From: Jeffrey Altman Date: Sat, 29 Jan 2011 23:24:16 -0500 Subject: [PATCH] Windows: out of order locks cm_CheckCBExpiration The recent refactoring of cm_CheckCBExpiration introduced a lock ordering error between the cm_scache_t rw lock and the cm_scacheLock. This patchset fixes the error by dropping the cm_scacheLock as each cm_scache_t is being processed. Change-Id: Ib9e45abc5a43ca550d4a2a7923e3b30017e9fbf9 Reviewed-on: http://gerrit.openafs.org/3773 Tested-by: Jeffrey Altman Reviewed-by: Jeffrey Altman --- src/WINNT/afsd/cm_callback.c | 28 ++++++++++++---------------- 1 file changed, 12 insertions(+), 16 deletions(-) diff --git a/src/WINNT/afsd/cm_callback.c b/src/WINNT/afsd/cm_callback.c index ac6a725280..dda6bf6613 100644 --- a/src/WINNT/afsd/cm_callback.c +++ b/src/WINNT/afsd/cm_callback.c @@ -1930,7 +1930,7 @@ void cm_CheckCBExpiration(void) { afs_uint32 i; cm_scache_t *scp; - cm_volume_t *volp = NULL; + cm_volume_t *volp; enum volstatus volstate; time_t now, downTime; @@ -1940,10 +1940,9 @@ void cm_CheckCBExpiration(void) lock_ObtainWrite(&cm_scacheLock); for (i=0; inextp) { - if (volp) { - cm_PutVolume(volp); - volp = NULL; - } + volp = NULL; + cm_HoldSCacheNoLock(scp); + lock_ReleaseWrite(&cm_scacheLock); /* * If this is not a PURERO object and there is no callback @@ -1951,7 +1950,7 @@ void cm_CheckCBExpiration(void) */ if (!(scp->flags & CM_SCACHEFLAG_PURERO) && (scp->cbServerp == NULL || scp->cbExpires == 0 || now < scp->cbExpires)) - continue; + goto scp_complete; /* * Determine the volume state and update the callback info @@ -1979,23 +1978,20 @@ void cm_CheckCBExpiration(void) /* If there is no callback or it hasn't expired, there is nothing to do */ if (scp->cbServerp == NULL || scp->cbExpires == 0 || now < scp->cbExpires) - continue; + goto scp_complete; /* If the volume is known not to be online, do not expire the callback */ if (volstate != vl_online) - continue; + goto scp_complete; /* * If all the servers are down and the callback expired after the * issuing server went down, do not expire the callback */ if (cm_CBServersDownTime(scp, volp, &downTime) && downTime && downTime < scp->cbExpires) - continue; + goto scp_complete; /* The callback has expired, discard the status info */ - cm_HoldSCacheNoLock(scp); - lock_ReleaseWrite(&cm_scacheLock); - osi_Log4(afsd_logp, "Callback Expiration Discarding SCache scp 0x%p vol %u vn %u uniq %u", scp, scp->fid.volume, scp->fid.vnode, scp->fid.unique); lock_ObtainWrite(&scp->rw); @@ -2004,14 +2000,14 @@ void cm_CheckCBExpiration(void) cm_CallbackNotifyChange(scp); + scp_complete: + if (volp) + cm_PutVolume(volp); + lock_ObtainWrite(&cm_scacheLock); cm_ReleaseSCacheNoLock(scp); } } - if (volp) { - cm_PutVolume(volp); - volp = NULL; - } lock_ReleaseWrite(&cm_scacheLock); osi_Log0(afsd_logp, "CheckCBExpiration Complete");