/* Editor Settings: expandtabs and use 4 spaces for indentation
* ex: set softtabstop=4 tabstop=8 expandtab shiftwidth=4: *
* -*- mode: c, c-basic-offset: 4 -*- */
/*
* Copyright Likewise Software
* All rights reserved.
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or (at
* your option) any later version.
*
* This program is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* for more details. You should have received a copy of the GNU General
* Public License along with this program. If not, see
* .
*
* LIKEWISE SOFTWARE MAKES THIS SOFTWARE AVAILABLE UNDER OTHER LICENSING
* TERMS AS WELL. IF YOU HAVE ENTERED INTO A SEPARATE LICENSE AGREEMENT
* WITH LIKEWISE SOFTWARE, THEN YOU MAY ELECT TO USE THE SOFTWARE UNDER THE
* TERMS OF THAT SOFTWARE LICENSE AGREEMENT INSTEAD OF THE TERMS OF THE GNU
* GENERAL PUBLIC LICENSE, NOTWITHSTANDING THE ABOVE NOTICE. IF YOU
* HAVE QUESTIONS, OR WISH TO REQUEST A COPY OF THE ALTERNATE LICENSING
* TERMS OFFERED BY LIKEWISE SOFTWARE, PLEASE CONTACT LIKEWISE SOFTWARE AT
* license@likewisesoftware.com
*/
/*
* Copyright (C) Likewise Software. All rights reserved.
*
* Module Name:
*
* lock.c
*
* Abstract:
*
* Likewise Posix File System Driver (PVFS)
*
* Lock/Unlock Dispatch Routine
*
* Authors: Gerald Carter
*/
#include "pvfs.h"
/* Forward declarations */
/* Code */
/**************************************************************
*************************************************************/
NTSTATUS
PvfsLockControl(
PPVFS_IRP_CONTEXT pIrpContext
)
{
NTSTATUS ntError = STATUS_UNSUCCESSFUL;
PIRP pIrp = pIrpContext->pIrp;
IRP_ARGS_LOCK_CONTROL Args = pIrp->Args.LockControl;
LONG64 Offset = Args.ByteOffset;
LONG64 Length = Args.Length;
ULONG Key = Args.Key;
PVFS_LOCK_FLAGS Flags = 0;
PPVFS_CCB pCcb = NULL;
PPVFS_PENDING_LOCK pLockCtx = NULL;
/* Sanity checks */
ntError = PvfsAcquireCCB(pIrp->FileHandle, &pCcb);
BAIL_ON_NT_STATUS(ntError);
if (PVFS_IS_DIR(pCcb)) {
ntError = STATUS_FILE_IS_A_DIRECTORY;
BAIL_ON_NT_STATUS(ntError);
}
/* Either READ or WRITE access is ok */
ntError = PvfsAccessCheckAnyFileHandle(
pCcb,
FILE_READ_DATA|FILE_WRITE_DATA);
BAIL_ON_NT_STATUS(ntError);
if (Args.ExclusiveLock) {
Flags |= PVFS_LOCK_EXCLUSIVE;
}
if (!Args.FailImmediately) {
Flags |= PVFS_LOCK_BLOCK;
}
switch(Args.LockControl)
{
case IO_LOCK_CONTROL_LOCK:
ntError = PvfsCreateLockContext(
&pLockCtx,
pIrpContext,
pCcb,
Key,
Offset,
Length,
Flags);
BAIL_ON_NT_STATUS(ntError);
ntError = PvfsOplockBreakIfLocked(pIrpContext, pCcb, pCcb->pFcb);
/* Question: What do we do for a lock that was specified as
FailImmediately? Should it block on an oplock break or
just immediately return failure? --jerry */
switch (ntError)
{
case STATUS_SUCCESS:
ntError = PvfsLockFileWithContext(pLockCtx);
break;
case STATUS_OPLOCK_BREAK_IN_PROGRESS:
ntError = PvfsPendOplockBreakTest(
pLockCtx->pCcb->pFcb,
pIrpContext,
pLockCtx->pCcb,
PvfsLockFileWithContext,
PvfsFreeLockContext,
(PVOID)pLockCtx);
if (ntError == STATUS_SUCCESS) {
pLockCtx = NULL;
ntError = STATUS_PENDING;
}
break;
case STATUS_PENDING:
ntError = PvfsAddItemPendingOplockBreakAck(
pLockCtx->pCcb->pFcb,
pIrpContext,
PvfsLockFileWithContext,
PvfsFreeLockContext,
(PVOID)pLockCtx);
if (ntError == STATUS_SUCCESS) {
pLockCtx = NULL;
ntError = STATUS_PENDING;
}
break;
default:
BAIL_ON_NT_STATUS(ntError);
break;
}
BAIL_ON_NT_STATUS(ntError);
break;
case IO_LOCK_CONTROL_UNLOCK:
ntError = PvfsUnlockFile(pCcb, FALSE, Key, Offset, Length);
break;
case IO_LOCK_CONTROL_UNLOCK_ALL_BY_KEY:
ntError = PvfsUnlockFile(pCcb, TRUE, Key, Offset, Length);
break;
case IO_LOCK_CONTROL_UNLOCK_ALL:
ntError = PvfsUnlockFile(pCcb, TRUE, 0, Offset, Length);
break;
default:
ntError = STATUS_INVALID_PARAMETER;
break;
}
BAIL_ON_NT_STATUS(ntError);
cleanup:
PvfsFreeLockContext((PVOID*)&pLockCtx);
if (pCcb)
{
PvfsReleaseCCB(pCcb);
}
return ntError;
error:
goto cleanup;
}
/*
local variables:
mode: c
c-basic-offset: 4
indent-tabs-mode: nil
tab-width: 4
end:
*/