|
|
/*++
Copyright (c) 1993 Microsoft Corporation
Module Name:
fileinfo.c
Abstract:
This module implements the get / set file information routines for Netware Redirector.
Author:
Manny Weiser (mannyw) 4-Mar-1993
Revision History:
--*/
#include "procs.h"
//
// The debug trace level
//
#define Dbg (DEBUG_TRACE_FILEINFO)
//
// local procedure prototypes
//
NTSTATUS NwCommonQueryInformation ( IN PIRP_CONTEXT pIrpContext );
NTSTATUS NwCommonSetInformation ( IN PIRP_CONTEXT pIrpContet );
NTSTATUS NwQueryBasicInfo ( IN PIRP_CONTEXT IrpContext, IN PICB Icb, IN PFILE_BASIC_INFORMATION Buffer );
NTSTATUS NwQueryStandardInfo ( IN PIRP_CONTEXT IrpContext, IN PICB Icb, IN PFILE_STANDARD_INFORMATION Buffer );
NTSTATUS NwQueryInternalInfo ( IN PIRP_CONTEXT IrpContext, IN PICB Icb, IN PFILE_INTERNAL_INFORMATION Buffer );
NTSTATUS NwQueryEaInfo ( IN PIRP_CONTEXT IrpContext, IN PFILE_EA_INFORMATION Buffer );
NTSTATUS NwQueryNameInfo ( IN PIRP_CONTEXT IrpContext, IN PICB Icb, IN PFILE_NAME_INFORMATION Buffer, IN OUT PULONG Length );
NTSTATUS NwQueryAltNameInfo ( IN PIRP_CONTEXT IrpContext, IN PICB Icb, IN PFILE_NAME_INFORMATION Buffer, IN OUT PULONG Length );
NTSTATUS NwQueryPositionInfo ( IN PIRP_CONTEXT IrpContext, IN PICB Icb, IN PFILE_POSITION_INFORMATION Buffer );
NTSTATUS NwSetBasicInfo ( IN PIRP_CONTEXT IrpContext, IN PICB Icb, IN PFILE_BASIC_INFORMATION Buffer );
NTSTATUS NwSetDispositionInfo ( IN PIRP_CONTEXT IrpContext, IN PICB Icb, IN PFILE_DISPOSITION_INFORMATION Buffer );
NTSTATUS NwSetRenameInfo ( IN PIRP_CONTEXT IrpContext, IN PICB Icb, IN PFILE_RENAME_INFORMATION Buffer );
NTSTATUS NwSetPositionInfo ( IN PIRP_CONTEXT IrpContext, IN PICB Icb, IN PFILE_POSITION_INFORMATION Buffer );
NTSTATUS NwSetAllocationInfo ( IN PIRP_CONTEXT IrpContext, IN PICB Icb, IN PFILE_ALLOCATION_INFORMATION Buffer );
NTSTATUS NwSetEndOfFileInfo ( IN PIRP_CONTEXT IrpContext, IN PICB Icb, IN PFILE_END_OF_FILE_INFORMATION Buffer );
#ifdef ALLOC_PRAGMA
#pragma alloc_text( PAGE, NwFsdQueryInformation )
#pragma alloc_text( PAGE, NwFsdSetInformation )
#pragma alloc_text( PAGE, NwCommonQueryInformation )
#pragma alloc_text( PAGE, NwCommonSetInformation )
#pragma alloc_text( PAGE, NwQueryStandardInfo )
#pragma alloc_text( PAGE, NwQueryInternalInfo )
#pragma alloc_text( PAGE, NwQueryEaInfo )
#pragma alloc_text( PAGE, NwQueryNameInfo )
#pragma alloc_text( PAGE, NwQueryPositionInfo )
#pragma alloc_text( PAGE, NwSetBasicInfo )
#pragma alloc_text( PAGE, NwSetDispositionInfo )
#pragma alloc_text( PAGE, NwDeleteFile )
#pragma alloc_text( PAGE, NwSetRenameInfo )
#pragma alloc_text( PAGE, NwSetPositionInfo )
#pragma alloc_text( PAGE, NwSetAllocationInfo )
#pragma alloc_text( PAGE, NwSetEndOfFileInfo )
#pragma alloc_text( PAGE, OccurenceCount )
#ifndef QFE_BUILD
#pragma alloc_text( PAGE1, NwQueryBasicInfo )
#endif
#endif
#if 0 // Not pageable
// see ifndef QFE_BUILD above
#endif
NTSTATUS NwFsdQueryInformation ( IN PDEVICE_OBJECT DeviceObject, IN PIRP Irp )
/*++
Routine Description:
This routine implements the FSD part of the NtQueryInformationFile API calls.
Arguments:
DeviceObject - Supplies a pointer to the device object to use.
Irp - Supplies a pointer to the Irp to process.
Return Value:
NTSTATUS - The Fsd status for the Irp
--*/
{ NTSTATUS status; PIRP_CONTEXT pIrpContext = NULL; BOOLEAN TopLevel;
PAGED_CODE();
DebugTrace(+1, Dbg, "NwFsdQueryInformation\n", 0);
//
// Call the common query information routine.
//
FsRtlEnterFileSystem(); TopLevel = NwIsIrpTopLevel( Irp );
try {
pIrpContext = AllocateIrpContext( Irp ); status = NwCommonQueryInformation( pIrpContext );
} except(NwExceptionFilter( Irp, GetExceptionInformation() )) {
if ( pIrpContext == NULL ) {
//
// If we couldn't allocate an irp context, just complete
// irp without any fanfare.
//
status = STATUS_INSUFFICIENT_RESOURCES; Irp->IoStatus.Status = status; Irp->IoStatus.Information = 0; IoCompleteRequest ( Irp, IO_NETWORK_INCREMENT );
} else {
//
// We had some trouble trying to perform the requested
// operation, so we'll abort the I/O request with
// the error Status that we get back from the
// execption code
//
status = NwProcessException( pIrpContext, GetExceptionCode() ); } }
if ( pIrpContext ) {
if ( status != STATUS_PENDING ) { NwDequeueIrpContext( pIrpContext, FALSE ); }
NwCompleteRequest( pIrpContext, status ); }
if ( TopLevel ) { NwSetTopLevelIrp( NULL ); } FsRtlExitFileSystem();
//
// Return to the caller.
//
DebugTrace(-1, Dbg, "NwFsdQueryInformation -> %08lx\n", status );
return status; }
NTSTATUS NwFsdSetInformation ( IN PDEVICE_OBJECT DeviceObject, IN PIRP Irp ) /*++
Routine Description:
This routine implements the FSD part of the NtSetInformationFile API calls.
Arguments:
DeviceObject - Supplies the device object to use.
Irp - Supplies the Irp being processed
Return Value:
NTSTATUS - The Fsd status for the Irp
--*/ { NTSTATUS status; PIRP_CONTEXT pIrpContext = NULL; BOOLEAN TopLevel;
PAGED_CODE();
DebugTrace(+1, Dbg, "NwFsdSetInformation\n", 0);
//
// Call the common Set Information routine.
//
FsRtlEnterFileSystem(); TopLevel = NwIsIrpTopLevel( Irp );
try {
pIrpContext = AllocateIrpContext( Irp ); status = NwCommonSetInformation( pIrpContext );
} except(NwExceptionFilter( Irp, GetExceptionInformation() )) {
if ( pIrpContext == NULL ) {
//
// If we couldn't allocate an irp context, just complete
// irp without any fanfare.
//
status = STATUS_INSUFFICIENT_RESOURCES; Irp->IoStatus.Status = status; Irp->IoStatus.Information = 0; IoCompleteRequest ( Irp, IO_NETWORK_INCREMENT );
} else {
//
// We had some trouble trying to perform the requested
// operation, so we'll abort the I/O request with
// the error Status that we get back from the
// execption code
//
status = NwProcessException( pIrpContext, GetExceptionCode() ); }
}
if ( pIrpContext ) {
if ( status != STATUS_PENDING ) { NwDequeueIrpContext( pIrpContext, FALSE ); }
NwCompleteRequest( pIrpContext, status ); }
if ( TopLevel ) { NwSetTopLevelIrp( NULL ); } FsRtlExitFileSystem();
//
// Return to the caller.
//
DebugTrace(-1, Dbg, "NwFsdSetInformation -> %08lx\n", status );
return status; }
NTSTATUS NwCommonQueryInformation ( IN PIRP_CONTEXT pIrpContext ) /*++
Routine Description:
This is the common routine for querying information on a file.
Arguments:
pIrpContext - Supplies Irp context information.
Return Value:
NTSTATUS - the return status for the operation.
--*/ { PIRP Irp; PIO_STACK_LOCATION irpSp; NTSTATUS status;
ULONG length; FILE_INFORMATION_CLASS fileInformationClass; PVOID buffer;
NODE_TYPE_CODE nodeTypeCode; PICB icb; PFCB fcb;
PVOID fsContext, fsContext2;
PFILE_ALL_INFORMATION AllInfo;
PAGED_CODE();
//
// Get the current stack location.
//
Irp = pIrpContext->pOriginalIrp; irpSp = IoGetCurrentIrpStackLocation( Irp );
DebugTrace(+1, Dbg, "NwCommonQueryInformation...\n", 0); DebugTrace( 0, Dbg, " Irp = %08lx\n", (ULONG_PTR)Irp); DebugTrace( 0, Dbg, " ->Length = %08lx\n", irpSp->Parameters.QueryFile.Length); DebugTrace( 0, Dbg, " ->FileInformationClass = %08lx\n", irpSp->Parameters.QueryFile.FileInformationClass); DebugTrace( 0, Dbg, " ->Buffer = %08lx\n", (ULONG_PTR)Irp->AssociatedIrp.SystemBuffer);
//
// Find out who are.
//
if ((nodeTypeCode = NwDecodeFileObject( irpSp->FileObject, &fsContext, &fsContext2 )) == NTC_UNDEFINED) {
status = STATUS_INVALID_HANDLE;
DebugTrace(-1, Dbg, "NwCommonQueryInformation -> %08lx\n", status ); return status; }
//
// Make sure that this the user is querying an ICB.
//
switch (nodeTypeCode) {
case NW_NTC_ICB:
icb = (PICB)fsContext2; break;
default: // This is an illegal file object to query
DebugTrace(0, Dbg, "Node type code is not incorrect\n", 0);
DebugTrace(-1, Dbg, "NwCommonQueryInformation -> STATUS_INVALID_PARAMETER\n", 0); return STATUS_INVALID_PARAMETER; }
pIrpContext->Icb = icb;
//
// Make local copies of the input parameters.
//
length = irpSp->Parameters.QueryFile.Length; fileInformationClass = irpSp->Parameters.QueryFile.FileInformationClass; buffer = Irp->AssociatedIrp.SystemBuffer;
//
// Now acquire shared access to the FCB
//
fcb = icb->SuperType.Fcb;
try {
NwVerifyIcbSpecial( icb );
//
// Based on the information class we'll do different actions. Each
// of the procedure that we're calling fill up as much of the
// buffer as possible and return the remaining length, and status
// This is done so that we can use them to build up the
// FileAllInformation request. These procedures do not complete the
// IRP, instead this procedure must complete the IRP.
//
status = STATUS_SUCCESS;
switch (fileInformationClass) {
case FileAllInformation:
AllInfo = buffer;
//
// First call all the Query Info handlers we can call
// synchronously.
//
NwQueryInternalInfo( pIrpContext, icb, &AllInfo->InternalInformation ); NwQueryEaInfo( pIrpContext, &AllInfo->EaInformation ); NwQueryPositionInfo( pIrpContext, icb, &AllInfo->PositionInformation );
length -= FIELD_OFFSET( FILE_ALL_INFORMATION, NameInformation );
status = NwQueryNameInfo( pIrpContext, icb, &AllInfo->NameInformation, &length );
if ( !NT_ERROR( status ) ) { status = NwQueryStandardInfo( pIrpContext, icb, &AllInfo->StandardInformation ); }
if ( !NT_ERROR( status ) ) { status = NwQueryBasicInfo( pIrpContext, icb, &AllInfo->BasicInformation ); }
break;
case FileBasicInformation:
length -= sizeof( FILE_BASIC_INFORMATION ); status = NwQueryBasicInfo( pIrpContext, icb, buffer );
break;
case FileStandardInformation:
//
// We will handle this call for information asynchronously.
// The callback routine will fill in the missing data, and
// complete the IRP.
//
// Remember the buffer length, and status to return.
//
length -= sizeof( FILE_STANDARD_INFORMATION ); status = NwQueryStandardInfo( pIrpContext, icb, buffer ); break;
case FileInternalInformation:
status = NwQueryInternalInfo( pIrpContext, icb, buffer ); length -= sizeof( FILE_INTERNAL_INFORMATION ); break;
case FileEaInformation:
status = NwQueryEaInfo( pIrpContext, buffer ); length -= sizeof( FILE_EA_INFORMATION ); break;
case FilePositionInformation:
status = NwQueryPositionInfo( pIrpContext, icb, buffer ); length -= sizeof( FILE_POSITION_INFORMATION ); break;
case FileNameInformation:
status = NwQueryNameInfo( pIrpContext, icb, buffer, &length ); break;
case FileAlternateNameInformation:
if (!DisableAltFileName) { status = NwQueryAltNameInfo( pIrpContext, icb, buffer, &length); break; }
default:
status = STATUS_INVALID_PARAMETER; break; }
//
// Set the information field to the number of bytes actually
// filled in and then complete the request. (This is
// irrelavent if the Query worker function returned
// STATUS_PENDING).
//
if ( status != STATUS_PENDING ) { Irp->IoStatus.Information = irpSp->Parameters.QueryFile.Length - length; }
} finally {
DebugTrace(-1, Dbg, "NwCommonQueryInformation -> %08lx\n", status ); }
return status; }
NTSTATUS NwCommonSetInformation ( IN PIRP_CONTEXT IrpContext ) /*++
Routine Description:
This is the common routine for setting information on a file.
Arguments:
IrpContext - Supplies the Irp to process
Return Value:
NTSTATUS - the return status for the operation
--*/ { PIRP irp; PIO_STACK_LOCATION irpSp; NTSTATUS status;
ULONG length; FILE_INFORMATION_CLASS fileInformationClass; PVOID buffer;
NODE_TYPE_CODE nodeTypeCode; PICB icb; PFCB fcb; PVOID fsContext;
//
// Get the current Irp stack location.
//
irp = IrpContext->pOriginalIrp; irpSp = IoGetCurrentIrpStackLocation( irp );
DebugTrace(+1, Dbg, "NwCommonSetInformation...\n", 0); DebugTrace( 0, Dbg, " Irp = %08lx\n", (ULONG_PTR)irp); DebugTrace( 0, Dbg, " ->Length = %08lx\n", irpSp->Parameters.SetFile.Length); DebugTrace( 0, Dbg, " ->FileInformationClass = %08lx\n", irpSp->Parameters.SetFile.FileInformationClass); DebugTrace( 0, Dbg, " ->Buffer = %08lx\n", (ULONG_PTR)irp->AssociatedIrp.SystemBuffer);
//
// Get a pointer to the FCB and ensure that this is a server side
// handler to a file.
//
if ((nodeTypeCode = NwDecodeFileObject( irpSp->FileObject, &fsContext, (PVOID *)&icb )) == NTC_UNDEFINED ) {
status = STATUS_INVALID_HANDLE;
DebugTrace(-1, Dbg, "NwCommonSetInformation -> %08lx\n", status ); return status; }
//
// Make sure that this the user is querying an ICB.
//
switch (nodeTypeCode) {
case NW_NTC_ICB:
fcb = icb->SuperType.Fcb; break;
default: // This is an illegal file object to query
DebugTrace(0, Dbg, "Node type code is not incorrect\n", 0);
DebugTrace(-1, Dbg, "NwCommonSetInformation -> STATUS_INVALID_PARAMETER\n", 0); return STATUS_INVALID_PARAMETER; }
IrpContext->Icb = icb;
//
// Make local copies of the input parameters.
//
length = irpSp->Parameters.SetFile.Length; fileInformationClass = irpSp->Parameters.SetFile.FileInformationClass; buffer = irp->AssociatedIrp.SystemBuffer;
try {
NwVerifyIcb( icb );
//
// Based on the information class we'll do different actions. Each
// procedure that we're calling will complete the request.
//
switch (fileInformationClass) {
case FileBasicInformation:
status = NwSetBasicInfo( IrpContext, icb, buffer ); break;
case FileDispositionInformation:
status = NwSetDispositionInfo( IrpContext, icb, buffer ); break;
case FileRenameInformation:
status = NwSetRenameInfo( IrpContext, icb, buffer ); break;
case FilePositionInformation:
status = NwSetPositionInfo( IrpContext, icb, buffer ); break;
case FileLinkInformation:
status = STATUS_INVALID_DEVICE_REQUEST; break;
case FileAllocationInformation:
status = NwSetAllocationInfo( IrpContext, icb, buffer ); break;
case FileEndOfFileInformation:
status = NwSetEndOfFileInfo( IrpContext, icb, buffer ); break;
default:
status = STATUS_INVALID_PARAMETER; break; }
} finally {
DebugTrace(-1, Dbg, "NwCommonSetInformation -> %08lx\n", status); }
return status; }
NTSTATUS NwQueryBasicInfo ( IN PIRP_CONTEXT IrpContext, IN PICB Icb, OUT PFILE_BASIC_INFORMATION Buffer ) /*++
Routine Description:
This routine performs the query basic information operation. This routine cannot be paged, it is called from QueryStandardInfoCallback.
Arguments:
Icb - Supplies a pointer the ICB for the file being querying.
Buffer - Supplies a pointer to the buffer where the information is to be returned.
Return Value:
VOID
--*/
{ PFCB Fcb; NTSTATUS Status; ULONG Attributes; USHORT CreationDate; USHORT CreationTime = DEFAULT_TIME; USHORT LastAccessDate; USHORT LastModifiedDate; USHORT LastModifiedTime; BOOLEAN FirstTime = TRUE;
DebugTrace(0, Dbg, "QueryBasicInfo...\n", 0);
//
// Zero out the buffer.
//
RtlZeroMemory( Buffer, sizeof(FILE_BASIC_INFORMATION) ); Fcb = Icb->SuperType.Fcb;
//
// It is ok to attempt a reconnect if this request fails with a
// connection error.
//
SetFlag( IrpContext->Flags, IRP_FLAG_RECONNECTABLE );
NwAcquireSharedFcb( Fcb->NonPagedFcb, TRUE );
//
// If we already know the file attributes, simply return them.
//
if ( FlagOn( Fcb->Flags, FCB_FLAGS_ATTRIBUTES_ARE_VALID ) ) {
//
// Set the various fields in the record
//
Buffer->CreationTime = NwDateTimeToNtTime( Fcb->CreationDate, Fcb->CreationTime );
Buffer->LastAccessTime = NwDateTimeToNtTime( Fcb->LastAccessDate, DEFAULT_TIME );
Buffer->LastWriteTime = NwDateTimeToNtTime( Fcb->LastModifiedDate, Fcb->LastModifiedTime );
Buffer->ChangeTime.QuadPart = 0;
DebugTrace(0, Dbg, "QueryBasic known %wZ\n", &Fcb->RelativeFileName); DebugTrace(0, Dbg, "LastModifiedDate %x\n", Fcb->LastModifiedDate); DebugTrace(0, Dbg, "LastModifiedTime %x\n", Fcb->LastModifiedTime); DebugTrace(0, Dbg, "CreationDate %x\n", Fcb->CreationDate ); DebugTrace(0, Dbg, "CreationTime %x\n", Fcb->CreationTime ); DebugTrace(0, Dbg, "LastAccessDate %x\n", Fcb->LastAccessDate );
Buffer->FileAttributes = Fcb->NonPagedFcb->Attributes;
if ( Buffer->FileAttributes == 0 ) { Buffer->FileAttributes = FILE_ATTRIBUTE_NORMAL; }
NwReleaseFcb( Fcb->NonPagedFcb ); return STATUS_SUCCESS;
} else if ( Fcb->RelativeFileName.Length == 0 ) {
//
// Allow 'cd \' to work.
//
Buffer->FileAttributes = FILE_ATTRIBUTE_DIRECTORY;
Buffer->CreationTime = NwDateTimeToNtTime( DEFAULT_DATE, DEFAULT_TIME );
Buffer->LastAccessTime = Buffer->CreationTime; Buffer->LastWriteTime = Buffer->CreationTime; Buffer->ChangeTime.QuadPart = 0;
NwReleaseFcb( Fcb->NonPagedFcb ); return STATUS_SUCCESS;
} else {
NwReleaseFcb( Fcb->NonPagedFcb );
IrpContext->pNpScb = Fcb->Scb->pNpScb; Retry: if ( !BooleanFlagOn( Fcb->Flags, FCB_FLAGS_LONG_NAME ) ) {
DebugTrace(0, Dbg, "QueryBasic short %wZ\n", &Fcb->RelativeFileName);
Status = ExchangeWithWait ( IrpContext, SynchronousResponseCallback, "FwbbJ", NCP_SEARCH_FILE, -1, Fcb->Vcb->Specific.Disk.Handle, Fcb->NodeTypeCode == NW_NTC_FCB ? SEARCH_ALL_FILES : SEARCH_ALL_DIRECTORIES, &Icb->SuperType.Fcb->RelativeFileName );
if ( NT_SUCCESS( Status ) ) {
Status = ParseResponse( IrpContext, IrpContext->rsp, IrpContext->ResponseLength, "N==_b-==wwww", 14, &Attributes, &CreationDate, &LastAccessDate, &LastModifiedDate, &LastModifiedTime);
//
// If this was a directory, there's no usable
// time/date info from the server.
//
if ( ( NT_SUCCESS( Status ) ) && ( Attributes & NW_ATTRIBUTE_DIRECTORY ) ) {
CreationDate = DEFAULT_DATE; LastAccessDate = DEFAULT_DATE; LastModifiedDate = DEFAULT_DATE; LastModifiedTime = DEFAULT_TIME;
} }
} else {
DebugTrace(0, Dbg, "QueryBasic long %wZ\n", &Fcb->RelativeFileName);
Status = ExchangeWithWait ( IrpContext, SynchronousResponseCallback, "LbbWDbDbC", NCP_LFN_GET_INFO, Fcb->Vcb->Specific.Disk.LongNameSpace, Fcb->Vcb->Specific.Disk.LongNameSpace, Fcb->NodeTypeCode == NW_NTC_FCB ? SEARCH_ALL_FILES : SEARCH_ALL_DIRECTORIES, LFN_FLAG_INFO_ATTRIBUTES | LFN_FLAG_INFO_MODIFY_TIME | LFN_FLAG_INFO_CREATION_TIME, Fcb->Vcb->Specific.Disk.VolumeNumber, Fcb->Vcb->Specific.Disk.Handle, 0, &Icb->SuperType.Fcb->RelativeFileName );
if ( NT_SUCCESS( Status ) ) { Status = ParseResponse( IrpContext, IrpContext->rsp, IrpContext->ResponseLength, "N_e_xx_xx_x", 4, &Attributes, 12, &CreationTime, &CreationDate, 4, &LastModifiedTime, &LastModifiedDate, 4, &LastAccessDate );
} }
if ( NT_SUCCESS( Status ) ) {
//
// Set the various fields in the record
//
Buffer->CreationTime = NwDateTimeToNtTime( CreationDate, CreationTime );
Buffer->LastAccessTime = NwDateTimeToNtTime( LastAccessDate, DEFAULT_TIME );
Buffer->LastWriteTime = NwDateTimeToNtTime( LastModifiedDate, LastModifiedTime );
Buffer->ChangeTime.QuadPart = 0;
DebugTrace(0, Dbg, "CreationDate %x\n", CreationDate ); DebugTrace(0, Dbg, "CreationTime %x\n", CreationTime ); DebugTrace(0, Dbg, "LastAccessDate %x\n", LastAccessDate ); DebugTrace(0, Dbg, "LastModifiedDate %x\n", LastModifiedDate); DebugTrace(0, Dbg, "LastModifiedTime %x\n", LastModifiedTime);
Buffer->FileAttributes = (UCHAR)Attributes;
if ( Buffer->FileAttributes == 0 ) { Buffer->FileAttributes = FILE_ATTRIBUTE_NORMAL; }
} else if ((Status == STATUS_INVALID_HANDLE) && (FirstTime)) {
//
// Check to see if Volume handle is invalid. Caused when volume
// is unmounted and then remounted.
//
FirstTime = FALSE;
NwReopenVcbHandle( IrpContext, Fcb->Vcb );
goto Retry; }
return( Status ); } }
#if NWFASTIO
BOOLEAN NwFastQueryBasicInfo ( IN PFILE_OBJECT FileObject, IN BOOLEAN Wait, IN OUT PFILE_BASIC_INFORMATION Buffer, OUT PIO_STATUS_BLOCK IoStatus, IN PDEVICE_OBJECT DeviceObject )
/*++
Routine Description:
This routine is for the fast query call for standard file information.
Arguments:
FileObject - Supplies the file object used in this operation
Wait - Indicates if we are allowed to wait for the information
Buffer - Supplies the output buffer to receive the basic information
IoStatus - Receives the final status of the operation
Return Value:
BOOLEAN - TRUE if the operation succeeded and FALSE if the caller needs to take the long route.
--*/
{ NODE_TYPE_CODE NodeTypeCode; PICB Icb; PFCB Fcb; PVOID FsContext;
FsRtlEnterFileSystem();
try { //
// Find out who are.
//
if ((NodeTypeCode = NwDecodeFileObject( FileObject, &FsContext, &Icb )) != NW_NTC_ICB ) { DebugTrace(-1, Dbg, "NwFastQueryStandardInfo -> FALSE\n", 0 ); return FALSE; } Fcb = Icb->SuperType.Fcb; NwAcquireExclusiveFcb( Fcb->NonPagedFcb, TRUE ); //
// If we don't have the info handy, we can't use the fast path.
//
if ( !FlagOn( Fcb->Flags, FCB_FLAGS_ATTRIBUTES_ARE_VALID ) ) { NwReleaseFcb( Fcb->NonPagedFcb ); return( FALSE ); } //
// Set the various fields in the record
//
Buffer->CreationTime = NwDateTimeToNtTime( Fcb->CreationDate, Fcb->CreationTime ); Buffer->LastAccessTime = NwDateTimeToNtTime( Fcb->LastAccessDate, DEFAULT_TIME ); Buffer->LastWriteTime = NwDateTimeToNtTime( Fcb->LastModifiedDate, Fcb->LastModifiedTime ); Buffer->ChangeTime.QuadPart = 0;
DebugTrace(0, Dbg, "QueryBasic known %wZ\n", &Fcb->RelativeFileName); DebugTrace(0, Dbg, "LastModifiedDate %x\n", Fcb->LastModifiedDate); DebugTrace(0, Dbg, "LastModifiedTime %x\n", Fcb->LastModifiedTime); DebugTrace(0, Dbg, "CreationDate %x\n", Fcb->CreationDate ); DebugTrace(0, Dbg, "CreationTime %x\n", Fcb->CreationTime ); DebugTrace(0, Dbg, "LastAccessDate %x\n", Fcb->LastAccessDate ); Buffer->FileAttributes = Fcb->NonPagedFcb->Attributes; if ( Buffer->FileAttributes == 0 ) { Buffer->FileAttributes = FILE_ATTRIBUTE_NORMAL; } IoStatus->Status = STATUS_SUCCESS; IoStatus->Information = sizeof( *Buffer ); NwReleaseFcb( Fcb->NonPagedFcb ); return TRUE; } finally { FsRtlExitFileSystem(); } } #endif
NTSTATUS NwQueryStandardInfo ( IN PIRP_CONTEXT IrpContext, IN PICB Icb, IN PFILE_STANDARD_INFORMATION Buffer )
/*++
Routine Description:
This routine perforNw the query standard information operation.
Arguments:
Fcb - Supplies the FCB of the being queried
Buffer - Supplies a pointer to the buffer where the information is to be returned
Return Value:
VOID
--*/
{ NTSTATUS Status; PFCB Fcb; ULONG FileSize; BOOLEAN FirstTime = TRUE;
PAGED_CODE();
Fcb = Icb->SuperType.Fcb;
//
// Zero out the buffer.
//
RtlZeroMemory( Buffer, sizeof(FILE_STANDARD_INFORMATION) );
//
// Fill in the answers we already know.
//
Buffer->NumberOfLinks = 1;
Buffer->DeletePending = (BOOLEAN)FlagOn( Fcb->Flags, FCB_FLAGS_DELETE_ON_CLOSE );
if ( Fcb->NodeTypeCode == NW_NTC_FCB ) { Buffer->Directory = FALSE; } else { Buffer->Directory = TRUE; }
if ( !Icb->HasRemoteHandle ) {
//
// It is ok to attempt a reconnect if this request fails with a
// connection error.
//
SetFlag( IrpContext->Flags, IRP_FLAG_RECONNECTABLE );
if ( Fcb->NodeTypeCode == NW_NTC_DCB || FlagOn( Fcb->Vcb->Flags, VCB_FLAG_PRINT_QUEUE ) ) {
//
// Allow 'cd \' to work.
//
Buffer->AllocationSize.QuadPart = 0; Buffer->EndOfFile.QuadPart = 0;
return STATUS_SUCCESS;
} else {
//
// No open handle for this file. Use a path based NCP
// to get the file size.
//
Retry: IrpContext->pNpScb = Fcb->Scb->pNpScb;
if ( !BooleanFlagOn( Icb->SuperType.Fcb->Flags, FCB_FLAGS_LONG_NAME ) ) {
Status = ExchangeWithWait ( IrpContext, SynchronousResponseCallback, "FwbbJ", NCP_SEARCH_FILE, -1, Fcb->Vcb->Specific.Disk.Handle, SEARCH_ALL_FILES, &Fcb->RelativeFileName );
if ( NT_SUCCESS( Status ) ) { Status = ParseResponse( IrpContext, IrpContext->rsp, IrpContext->ResponseLength, "N_d", 20, &FileSize ); }
} else {
Status = ExchangeWithWait ( IrpContext, SynchronousResponseCallback, "LbbWDbDbC", NCP_LFN_GET_INFO, Fcb->Vcb->Specific.Disk.LongNameSpace, Fcb->Vcb->Specific.Disk.LongNameSpace, SEARCH_ALL_FILES, LFN_FLAG_INFO_FILE_SIZE, Fcb->Vcb->Specific.Disk.VolumeNumber, Fcb->Vcb->Specific.Disk.Handle, 0, &Fcb->RelativeFileName );
if ( NT_SUCCESS( Status ) ) { Status = ParseResponse( IrpContext, IrpContext->rsp, IrpContext->ResponseLength, "N_e", 10, &FileSize ); }
}
if ((Status == STATUS_INVALID_HANDLE) && (FirstTime)) {
//
// Check to see if Volume handle is invalid. Caused when volume
// is unmounted and then remounted.
//
FirstTime = FALSE;
NwReopenVcbHandle( IrpContext, Fcb->Vcb );
goto Retry; }
Buffer->AllocationSize.QuadPart = FileSize; Buffer->EndOfFile.QuadPart = FileSize;
}
} else {
//
// Start a Get file size NCP
//
IrpContext->pNpScb = Fcb->Scb->pNpScb;
if ( Fcb->NodeTypeCode == NW_NTC_FCB ) { AcquireFcbAndFlushCache( IrpContext, Fcb->NonPagedFcb ); }
Status = ExchangeWithWait( IrpContext, SynchronousResponseCallback, "F-r", NCP_GET_FILE_SIZE, &Icb->Handle, sizeof(Icb->Handle ) );
if ( NT_SUCCESS( Status ) ) { //
// Get the data from the response.
//
Status = ParseResponse( IrpContext, IrpContext->rsp, IrpContext->ResponseLength, "Nd", &FileSize );
}
if ( NT_SUCCESS( Status ) ) {
//
// Fill in Allocation size and EOF, based on the response.
//
Buffer->AllocationSize.QuadPart = FileSize; Buffer->EndOfFile.QuadPart = Buffer->AllocationSize.QuadPart;
} }
return( Status ); }
#if NWFASTIO
BOOLEAN NwFastQueryStandardInfo ( IN PFILE_OBJECT FileObject, IN BOOLEAN Wait, IN OUT PFILE_STANDARD_INFORMATION Buffer, OUT PIO_STATUS_BLOCK IoStatus, IN PDEVICE_OBJECT DeviceObject ) /*++
Routine Description:
This routine is for the fast query call for standard file information.
Arguments:
FileObject - Supplies the file object used in this operation
Wait - Indicates if we are allowed to wait for the information
Buffer - Supplies the output buffer to receive the basic information
IoStatus - Receives the final status of the operation
Return Value:
BOOLEAN - TRUE if the operation succeeded and FALSE if the caller needs to take the long route.
--*/ { NODE_TYPE_CODE NodeTypeCode; PICB Icb; PFCB Fcb; PVOID FsContext;
//
// Find out who are.
//
try { FsRtlEnterFileSystem();
if ((NodeTypeCode = NwDecodeFileObject( FileObject, &FsContext, &Icb )) != NW_NTC_ICB ) { DebugTrace(-1, Dbg, "NwFastQueryStandardInfo -> FALSE\n", 0 ); return FALSE; } Fcb = Icb->SuperType.Fcb; //
// If we have the info handy, we can use the fast path.
//
if ( Fcb->NodeTypeCode == NW_NTC_DCB || FlagOn( Fcb->Vcb->Flags, VCB_FLAG_PRINT_QUEUE ) ) { Buffer->AllocationSize.QuadPart = 0; Buffer->EndOfFile.QuadPart = 0; Buffer->NumberOfLinks = 1; Buffer->DeletePending = (BOOLEAN)FlagOn( Fcb->Flags, FCB_FLAGS_DELETE_ON_CLOSE ); Buffer->Directory = TRUE; IoStatus->Status = STATUS_SUCCESS; IoStatus->Information = sizeof( *Buffer ); return TRUE; } else { return FALSE; } } finally {
FsRtlExitFileSystem(); } } #endif
NTSTATUS NwQueryInternalInfo ( IN PIRP_CONTEXT IrpContext, IN PICB Icb, IN PFILE_INTERNAL_INFORMATION Buffer )
/*++
Routine Description:
This routine perforNw the query internal information operation.
Arguments:
Fcb - Supplies the FCB of the being queried.
Buffer - Supplies a pointer to the buffer where the information is to be returned.
Return Value:
VOID
--*/
{ PAGED_CODE();
DebugTrace(0, Dbg, "QueryInternalInfo...\n", 0);
//
// Zero out the buffer.
//
RtlZeroMemory( Buffer, sizeof(FILE_INTERNAL_INFORMATION) );
//
// Set the internal index number to be the address of the ICB.
//
Buffer->IndexNumber.HighPart = 0; Buffer->IndexNumber.QuadPart = (ULONG_PTR)Icb->NpFcb;
return( STATUS_SUCCESS ); }
NTSTATUS NwQueryEaInfo ( IN PIRP_CONTEXT IrpContext, IN PFILE_EA_INFORMATION Buffer )
/*++
Routine Description:
This routine performs the query Ea information operation.
Arguments:
Buffer - Supplies a pointer to the buffer where the information is to be returned
Return Value:
VOID - The result of this query
--*/
{ PAGED_CODE();
DebugTrace(0, Dbg, "QueryEaInfo...\n", 0);
//
// Zero out the buffer.
//
RtlZeroMemory(Buffer, sizeof(FILE_EA_INFORMATION));
return STATUS_SUCCESS; }
NTSTATUS NwQueryNameInfo ( IN PIRP_CONTEXT IrpContext, IN PICB Icb, IN PFILE_NAME_INFORMATION Buffer, IN PULONG Length )
/*++
Routine Description:
This routine performs the query name information operation.
Arguments:
Fcb - Supplies the FCB of the file to query.
Buffer - Supplies a pointer to the buffer where the information is to be returned
Length - Supplies and receives the length of the buffer in bytes.
Return Value:
NTSTATUS - The result of this query.
--*/
{ ULONG bytesToCopy; ULONG fileNameSize; PFCB Fcb = Icb->SuperType.Fcb;
NTSTATUS status;
PAGED_CODE();
DebugTrace(0, Dbg, "QueryNameInfo...\n", 0);
//
// Win32 expects the root directory name to be '\' terminated,
// the netware server does not. So if this is a root directory,
// (i.e RelativeFileName length is 0) append a '\' to the path name.
//
//
// See if the buffer is large enough, and decide how many bytes to copy.
//
*Length -= FIELD_OFFSET( FILE_NAME_INFORMATION, FileName[0] );
fileNameSize = Fcb->FullFileName.Length; if ( Fcb->RelativeFileName.Length == 0 ) { fileNameSize += sizeof(L'\\'); } Buffer->FileNameLength = fileNameSize;
if ( *Length >= fileNameSize ) {
status = STATUS_SUCCESS;
bytesToCopy = fileNameSize;
} else {
status = STATUS_BUFFER_OVERFLOW;
bytesToCopy = *Length; }
//
// Copy over the file name and its length.
//
RtlMoveMemory( Buffer->FileName, Fcb->FullFileName.Buffer, bytesToCopy);
//
// If this is a root directory, and there is space in the buffer
// append a '\' to make win32 happy.
//
if ( Fcb->RelativeFileName.Length == 0 && status == STATUS_SUCCESS ) { Buffer->FileName[ fileNameSize/sizeof(WCHAR) - 1 ] = L'\\'; }
*Length -= bytesToCopy;
return status; }
NTSTATUS NwQueryAltNameInfo ( IN PIRP_CONTEXT pIrpContext, IN PICB Icb, IN PFILE_NAME_INFORMATION Buffer, IN PULONG Length )
/*++
Routine Description:
This routine performs the AltName query name information operation.
Arguments:
Fcb - Supplies the FCB of the file to query.
Buffer - Supplies a pointer to the buffer where the information is to be returned
Length - Supplies and receives the length of the buffer in bytes.
Return Value:
NTSTATUS - The result of this query.
--*/
{ ULONG bytesToCopy; ULONG fileNameSize; PFCB Fcb = Icb->SuperType.Fcb;
UNICODE_STRING ShortName; NTSTATUS status;
PAGED_CODE();
DebugTrace(0, Dbg, "QueryAltNameInfo...\n", 0);
pIrpContext->pNpScb = Fcb->Scb->pNpScb;
//
// See if the buffer is large enough, and decide how many bytes to copy.
//
*Length -= FIELD_OFFSET( FILE_NAME_INFORMATION, FileName[0] );
ShortName.MaximumLength = MAX_PATH; ShortName.Buffer=NULL; ShortName.Length = 0;
status = ExchangeWithWait ( pIrpContext, SynchronousResponseCallback, "LbbWDbDbC", NCP_LFN_GET_INFO, Fcb->Vcb->Specific.Disk.LongNameSpace, 0x0, //0x0 DOS Nam
SEARCH_ALL_DIRECTORIES, LFN_FLAG_INFO_NAME, Fcb->Vcb->Specific.Disk.VolumeNumber, Fcb->Vcb->Specific.Disk.Handle, 0, &Fcb->RelativeFileName );
if (!NT_SUCCESS( status ) ){ return status; }
ShortName.Buffer= ALLOCATE_POOL(NonPagedPool, ShortName.MaximumLength+sizeof(WCHAR)); if (ShortName.Buffer == NULL){ return STATUS_INSUFFICIENT_RESOURCES; }
status = ParseResponse( pIrpContext, pIrpContext->rsp, pIrpContext->ResponseLength, "N_P", 76, &ShortName);
if ( NT_SUCCESS( status ) ) {
fileNameSize = ShortName.Length;
if ( *Length >= fileNameSize ) {
status = STATUS_SUCCESS; bytesToCopy = fileNameSize;
} else {
status = STATUS_BUFFER_OVERFLOW; bytesToCopy = *Length; }
Buffer->FileNameLength = fileNameSize;
RtlMoveMemory( Buffer->FileName, ShortName.Buffer, bytesToCopy);
*Length -= bytesToCopy;
}
FREE_POOL(ShortName.Buffer);
return status; }
NTSTATUS NwQueryPositionInfo ( IN PIRP_CONTEXT IrpContext, IN PICB Icb, IN PFILE_POSITION_INFORMATION Buffer )
/*++
Routine Description:
This routine performs the query position information operation.
Arguments:
Fcb - Supplies the FCB of the file being queried.
Buffer - Supplies a pointer to the buffer where the information is to be returned.
Return Value:
VOID
--*/
{ PAGED_CODE();
DebugTrace(0, Dbg, "QueryPositionInfo...\n", 0);
//
// Return the current byte offset. This info is totally
// bogus for asynchronous files. Also note that we don't
// use the FilePosition member of the ICB for anything.
//
if ( Icb->FileObject ) { Buffer->CurrentByteOffset.QuadPart = Icb->FileObject->CurrentByteOffset.QuadPart; }
return STATUS_SUCCESS; }
NTSTATUS NwSetBasicInfo ( IN PIRP_CONTEXT pIrpContext, IN PICB Icb, IN PFILE_BASIC_INFORMATION Buffer ) /*++
Routine Description:
This routine sets the basic information for a file.
Arguments:
pIrpContext - Supplies Irp context information.
Icb - Supplies the ICB for the file being modified.
Buffer - Supplies the buffer containing the data being set.
Return Value:
NTSTATUS - Returns our completion status.
--*/
{ PFCB Fcb; NTSTATUS Status; BOOLEAN SetTime = FALSE; BOOLEAN SetAttributes = FALSE; ULONG LfnFlag = 0;
PAGED_CODE();
DebugTrace(0, Dbg, "SetBasicInfo...\n", 0);
Fcb = Icb->SuperType.Fcb;
pIrpContext->pNpScb = Fcb->Scb->pNpScb;
//
// Append this IRP context and wait to get to the front.
// then grab from FCB
//
NwAppendToQueueAndWait( pIrpContext ); NwAcquireExclusiveFcb( Fcb->NonPagedFcb, TRUE );
//
// It is ok to attempt a reconnect if this request fails with a
// connection error.
//
SetFlag( pIrpContext->Flags, IRP_FLAG_RECONNECTABLE );
if (Buffer->CreationTime.QuadPart != 0) {
//
// Modify the creation time.
//
Status = NwNtTimeToNwDateTime( Buffer->CreationTime, &Fcb->CreationDate, &Fcb->CreationTime );
if ( !NT_SUCCESS( Status ) ) { NwReleaseFcb( Fcb->NonPagedFcb ); return( Status ); }
SetTime = TRUE; LfnFlag |= LFN_FLAG_SET_INFO_CREATE_DATE | LFN_FLAG_SET_INFO_CREATE_TIME; }
if (Buffer->LastAccessTime.QuadPart != 0) {
USHORT Dummy;
//
// Modify the last access time.
//
Status = NwNtTimeToNwDateTime( Buffer->LastAccessTime, &Fcb->LastAccessDate, &Dummy );
if ( !NT_SUCCESS( Status ) ) { NwReleaseFcb( Fcb->NonPagedFcb ); return( Status ); }
SetTime = TRUE; LfnFlag |= LFN_FLAG_SET_INFO_LASTACCESS_DATE;
// Set the last access flag in the ICB so that we update
// last access time for real when we close this handle!
Icb->UserSetLastAccessTime = TRUE; }
if (Buffer->LastWriteTime.QuadPart != 0) {
//
// Modify the last write time
//
Status = NwNtTimeToNwDateTime( Buffer->LastWriteTime, &Fcb->LastModifiedDate, &Fcb->LastModifiedTime );
if ( !NT_SUCCESS( Status ) ) { NwReleaseFcb( Fcb->NonPagedFcb ); return( Status ); }
LfnFlag |= LFN_FLAG_SET_INFO_MODIFY_DATE | LFN_FLAG_SET_INFO_MODIFY_TIME; }
if (Buffer->FileAttributes != 0) { LfnFlag |= LFN_FLAG_SET_INFO_ATTRIBUTES; }
if ( LfnFlag == 0 ) {
//
// Nothing to set, simply return success.
//
Status = STATUS_SUCCESS; }
if ( Fcb->NodeTypeCode == NW_NTC_FCB ) {
//
// Call plain FlushCache - we don't want to acquire and
// release the NpFcb. We are already at the front and have the Fcb
// exclusive.
//
FlushCache( pIrpContext, Fcb->NonPagedFcb ); }
if ( BooleanFlagOn( Fcb->Flags, FCB_FLAGS_LONG_NAME ) ) {
Status = ExchangeWithWait( pIrpContext, SynchronousResponseCallback, "LbbWDW--WW==WW==_W_bDbC", NCP_LFN_SET_INFO, Fcb->Vcb->Specific.Disk.LongNameSpace, Fcb->Vcb->Specific.Disk.LongNameSpace, Fcb->NodeTypeCode == NW_NTC_FCB ? SEARCH_ALL_FILES : SEARCH_ALL_DIRECTORIES, LfnFlag, NtAttributesToNwAttributes( Buffer->FileAttributes ), Fcb->CreationDate, Fcb->CreationTime, Fcb->LastModifiedDate, Fcb->LastModifiedTime, 8, Fcb->LastAccessDate, 8, Fcb->Vcb->Specific.Disk.VolumeNumber, Fcb->Vcb->Specific.Disk.Handle, 0, &Fcb->RelativeFileName );
} else {
if ( LfnFlag & LFN_FLAG_SET_INFO_ATTRIBUTES ) { Status = ExchangeWithWait( pIrpContext, SynchronousResponseCallback, "FbbbU", NCP_SET_FILE_ATTRIBUTES, NtAttributesToNwAttributes( Buffer->FileAttributes ), Fcb->Vcb->Specific.Disk.Handle, Fcb->NodeTypeCode == NW_NTC_FCB ? SEARCH_ALL_FILES : SEARCH_ALL_DIRECTORIES, &Fcb->RelativeFileName );
if ( !NT_SUCCESS( Status ) ) { NwReleaseFcb( Fcb->NonPagedFcb ); return( Status ); }
}
#if 0
//
// We could conceivably use ScanDir/SetDir to update last access
// and create time. Not supported yet.
//
if ( LfnFlag & ( LFN_FLAG_SET_INFO_LASTACCESS_DATE | LFN_FLAG_SET_INFO_CREATE_DATE ) ) {
ULONG SearchIndex; ULONG Directory;
Status = ExchangeWithWait( pIrpContext, SynchronousResponseCallback, "SbbdU", 0x16, 0x1E, // Scan dir entry
Fcb->Vcb->Specific.Disk.Handle, 0x06, // Search attributes
-1, // Search index
&Fcb->RelativeFileName );
if ( NT_SUCCESS( Status ) ) { Status = ParseResponse( pIrpContext, pIrpContext->rsp, pIrpContext->ResponseLength, "Ndd", &SearchIndex, &Directory ); }
if ( NT_SUCCESS( Status ) ) { Status = ExchangeWithWait( pIrpContext, SynchronousResponseCallback, "Sbbdddw=----_ww==ww==ww", 0x16, 0x25, // Set dir entry
Fcb->Vcb->Specific.Disk.Handle, 0x06, // Search attributes
SearchIndex, 0, // Change Bits?
Directory, 12, Fcb->CreationDate, 0, Fcb->LastAccessDate, 0, Fcb->LastModifiedDate, Fcb->LastModifiedTime ); } } #endif
if ( LfnFlag & LFN_FLAG_SET_INFO_MODIFY_DATE ) { Status = ExchangeWithWait( pIrpContext, SynchronousResponseCallback, "F-rww-", NCP_SET_FILE_TIME, &Icb->Handle, sizeof( Icb->Handle ), Fcb->LastModifiedTime, Fcb->LastModifiedDate ); } }
NwReleaseFcb( Fcb->NonPagedFcb );
//
// And return to our caller
//
return Status; }
NTSTATUS NwSetDispositionInfo ( IN PIRP_CONTEXT pIrpContext, IN PICB Icb, IN PFILE_DISPOSITION_INFORMATION Buffer ) /*++
Routine Description:
This routine sets the disposition information for a file.
Arguments:
pIrpContext - Supplies Irp context information.
Icb - Supplies the ICB for the file being modified.
Buffer - Supplies the buffer containing the data being set.
Return Value:
NTSTATUS - Returns our completion status.
--*/ { PFCB Fcb; NTSTATUS Status;
PAGED_CODE();
DebugTrace(0, Dbg, "SetDispositionInfo...\n", 0);
Fcb = Icb->SuperType.Fcb;
if ( FlagOn( Fcb->Vcb->Flags, VCB_FLAG_PRINT_QUEUE ) ) {
//
// This is a print queue, just pretend this IRP succeeded.
//
Status = STATUS_SUCCESS;
} else {
//
// This is a real file or directory. Mark it delete pending.
//
SetFlag( Fcb->Flags, FCB_FLAGS_DELETE_ON_CLOSE );
pIrpContext->pNpScb = Fcb->Scb->pNpScb; pIrpContext->Icb = Icb;
Icb->State = ICB_STATE_CLOSE_PENDING;
//
// Go ahead, delete the file.
//
Status = NwDeleteFile( pIrpContext ); }
return( Status ); }
NTSTATUS NwDeleteFile( PIRP_CONTEXT pIrpContext ) /*++
Routine Description:
This routine continues processing of the SetDispositionInfo request. It must run in the redirector FSP.
Arguments:
pIrpContext - A pointer to the IRP context information for the request in progress.
Return Value:
The status of the operation.
--*/ { PICB Icb; PFCB Fcb; NTSTATUS Status;
PAGED_CODE();
Icb = pIrpContext->Icb; Fcb = Icb->SuperType.Fcb;
ClearFlag( Fcb->Flags, FCB_FLAGS_DELETE_ON_CLOSE );
//
// To a delete a file, first close the remote handle.
//
if ( Icb->HasRemoteHandle ) {
Icb->HasRemoteHandle = FALSE;
Status = ExchangeWithWait( pIrpContext, SynchronousResponseCallback, "F-r", NCP_CLOSE, Icb->Handle, sizeof( Icb->Handle ) ); }
//
// Note that this request cannot be reconnectable since, it can
// be called via NwCloseIcb(). See comment in that routine for
// more info.
//
if ( Fcb->NodeTypeCode == NW_NTC_FCB ) {
if ( !BooleanFlagOn( Fcb->Flags, FCB_FLAGS_LONG_NAME ) ) {
Status = ExchangeWithWait( pIrpContext, SynchronousResponseCallback, "FbbJ", NCP_DELETE_FILE, Fcb->Vcb->Specific.Disk.Handle, SEARCH_ALL_FILES, &Fcb->RelativeFileName );
} else {
Status = ExchangeWithWait( pIrpContext, SynchronousResponseCallback, "LbbW-DbC", NCP_LFN_DELETE_FILE, Fcb->Vcb->Specific.Disk.LongNameSpace, Fcb->Vcb->Specific.Disk.VolumeNumber, NW_ATTRIBUTE_SYSTEM | NW_ATTRIBUTE_HIDDEN, Fcb->Vcb->Specific.Disk.Handle, LFN_FLAG_SHORT_DIRECTORY, &Fcb->RelativeFileName ); }
} else {
ASSERT( Fcb->NodeTypeCode == NW_NTC_DCB );
if ( !BooleanFlagOn( Fcb->Flags, FCB_FLAGS_LONG_NAME ) ) {
Status = ExchangeWithWait( pIrpContext, SynchronousResponseCallback, "SbbJ", NCP_DIR_FUNCTION, NCP_DELETE_DIRECTORY, Fcb->Vcb->Specific.Disk.Handle, SEARCH_ALL_DIRECTORIES, &Fcb->RelativeFileName ); } else {
Status = ExchangeWithWait( pIrpContext, SynchronousResponseCallback, "LbbW-DbC", NCP_LFN_DELETE_FILE, Fcb->Vcb->Specific.Disk.LongNameSpace, Fcb->Vcb->Specific.Disk.VolumeNumber, SEARCH_ALL_DIRECTORIES, Fcb->Vcb->Specific.Disk.Handle, LFN_FLAG_SHORT_DIRECTORY, &Fcb->RelativeFileName ); }
}
if ( NT_SUCCESS( Status )) {
Status = ParseResponse( pIrpContext, pIrpContext->rsp, pIrpContext->ResponseLength, "N" );
} else {
//
// We can map all failures to STATUS_NO_SUCH_FILE
// except ACCESS_DENIED, which happens with a read
// only file.
//
if ( Status != STATUS_ACCESS_DENIED ) { Status = STATUS_NO_SUCH_FILE; }
}
return Status; }
NTSTATUS NwSetRenameInfo ( IN PIRP_CONTEXT IrpContext, IN PICB Icb, IN PFILE_RENAME_INFORMATION Buffer ) /*++
Routine Description:
This routine set rename information for a file.
Arguments:
pIrpContext - A pointer to the IRP context information for the request in progress.
Icb - A pointer to the ICB of the file to set.
Buffer - The request buffer.
Return Value:
The status of the operation.
--*/ { PIRP Irp; PIO_STACK_LOCATION irpSp; NTSTATUS Status; NTSTATUS Status2; PFCB Fcb; PFCB TargetFcb; BOOLEAN HandleAllocated = FALSE; BYTE Handle; PICB TargetIcb = NULL;
UNICODE_STRING OldDrive; UNICODE_STRING OldServer; UNICODE_STRING OldVolume; UNICODE_STRING OldPath; UNICODE_STRING OldFileName; UNICODE_STRING OldFullName; WCHAR OldDriveLetter; UNICODE_STRING OldFcbFullName;
UNICODE_STRING NewDrive; UNICODE_STRING NewServer; UNICODE_STRING NewVolume; UNICODE_STRING NewPath; UNICODE_STRING NewFileName; UNICODE_STRING NewFullName; WCHAR NewDriveLetter; UNICODE_STRING NewFcbFullName;
USHORT i;
PAGED_CODE();
DebugTrace(+1, Dbg, "SetRenameInfo...\n", 0);
//
// Can't try to set rename info on a print queue.
//
Fcb = Icb->SuperType.Fcb;
if ( FlagOn( Fcb->Vcb->Flags, VCB_FLAG_PRINT_QUEUE ) ) { return( STATUS_INVALID_PARAMETER ); }
//
// It is ok to attempt a reconnect if this request fails with a
// connection error.
//
SetFlag( IrpContext->Flags, IRP_FLAG_RECONNECTABLE );
//
// Get the current stack location.
//
Irp = IrpContext->pOriginalIrp; irpSp = IoGetCurrentIrpStackLocation( Irp );
DebugTrace( 0, Dbg, " ->FullFileName = %wZ\n", &Fcb->FullFileName);
if (irpSp->Parameters.SetFile.FileObject != NULL) {
TargetIcb = irpSp->Parameters.SetFile.FileObject->FsContext2;
DebugTrace( 0, Dbg, " ->FullFileName = %wZ\n", &TargetIcb->SuperType.Fcb->FullFileName);
if ( TargetIcb->SuperType.Fcb->Scb != Icb->SuperType.Fcb->Scb ) { return STATUS_NOT_SAME_DEVICE; }
} else {
DebugTrace( 0, Dbg, " ->FullFileName in users buffer\n", 0); DebugTrace(-1, Dbg, "SetRenameInfo %08lx\n", STATUS_NOT_IMPLEMENTED); return STATUS_NOT_IMPLEMENTED; }
DebugTrace( 0, Dbg, " ->TargetFileName = %wZ\n", &irpSp->Parameters.SetFile.FileObject->FileName);
TargetFcb = ((PNONPAGED_FCB)irpSp->Parameters.SetFile.FileObject->FsContext)->Fcb;
IrpContext->pNpScb = Fcb->Scb->pNpScb;
NwAppendToQueueAndWait( IrpContext ); NwAcquireExclusiveFcb( Fcb->NonPagedFcb, TRUE );
try {
//
// If either source or destination is a long name, use
// the long name path.
//
if ( !BooleanFlagOn( Fcb->Flags, FCB_FLAGS_LONG_NAME ) && IsFatNameValid( &TargetFcb->RelativeFileName ) && !BooleanFlagOn( Fcb->Vcb->Flags, VCB_FLAG_LONG_NAME ) ) {
//
// Strip to UID portion of the FCB name.
//
for ( i = 0 ; i < Fcb->FullFileName.Length / sizeof(WCHAR) ; i++ ) { if ( Fcb->FullFileName.Buffer[i] == OBJ_NAME_PATH_SEPARATOR ) { break; } }
ASSERT( Fcb->FullFileName.Buffer[i] == OBJ_NAME_PATH_SEPARATOR );
OldFcbFullName.Length = Fcb->FullFileName.Length - i*sizeof(WCHAR); OldFcbFullName.Buffer = Fcb->FullFileName.Buffer + i;
Status = CrackPath ( &OldFcbFullName, &OldDrive, &OldDriveLetter, &OldServer, &OldVolume, &OldPath, &OldFileName, &OldFullName );
ASSERT(NT_SUCCESS(Status));
//
// Strip to UID portion of the FCB name.
//
TargetFcb = ((PNONPAGED_FCB)(irpSp->Parameters.SetFile.FileObject->FsContext))->Fcb;
for ( i = 0 ; i < TargetFcb->FullFileName.Length / sizeof(WCHAR) ; i++ ) { if ( TargetFcb->FullFileName.Buffer[i] == OBJ_NAME_PATH_SEPARATOR ) { break; } }
ASSERT( TargetFcb->FullFileName.Buffer[i] == OBJ_NAME_PATH_SEPARATOR );
NewFcbFullName.Length = TargetFcb->FullFileName.Length - i*sizeof(WCHAR); NewFcbFullName.Buffer = TargetFcb->FullFileName.Buffer + i;
Status = CrackPath ( &NewFcbFullName, &NewDrive, &NewDriveLetter, &NewServer, &NewVolume, &NewPath, &NewFileName, &NewFullName );
ASSERT(NT_SUCCESS(Status));
//
// Make sure that this is the same volume.
//
if ( RtlCompareUnicodeString( &NewVolume, &OldVolume, TRUE ) != 0 ) { try_return( Status = STATUS_NOT_SAME_DEVICE ); }
if (Icb->SuperType.Fcb->IcbCount != 1) { try_return( Status = STATUS_ACCESS_DENIED ); }
//
// After a rename, the only operation allowed on the handle is an
// NtClose.
//
Icb->State = ICB_STATE_CLOSE_PENDING;
if ((irpSp->Parameters.SetFile.ReplaceIfExists ) && (TargetIcb->Exists)) {
// Delete the file
Status2 = ExchangeWithWait( IrpContext, SynchronousResponseCallback, "Fb-J", NCP_DELETE_FILE, TargetFcb->Vcb->Specific.Disk.Handle, &TargetFcb->RelativeFileName );
#ifdef NWDBG
if ( NT_SUCCESS( Status2 ) ) { Status2 = ParseResponse( IrpContext, IrpContext->rsp, IrpContext->ResponseLength, "N" ); }
ASSERT(NT_SUCCESS(Status2)); #endif
}
//
// Need to create a handle to the directory containing the old
// file/directory name because directory rename does not contain a
// path and there might not be room for two paths in a file rename.
//
// The way we do this is to allocate a temporary handle on the server.
// This request is at the front of the Scb->Requests queue and so can
// use the temporary handle and delete it without affecting any other
// requests.
//
if ( OldPath.Length == 0 ) {
// In the root so use the VCB handle.
Handle = Fcb->Vcb->Specific.Disk.Handle;
} else {
Status = ExchangeWithWait ( IrpContext, SynchronousResponseCallback, "SbbJ", // NCP Allocate temporary directory handle
NCP_DIR_FUNCTION, NCP_ALLOCATE_TEMP_DIR_HANDLE, Fcb->Vcb->Specific.Disk.Handle, '[', &OldPath );
if ( NT_SUCCESS( Status ) ) { Status = ParseResponse( IrpContext, IrpContext->rsp, IrpContext->ResponseLength, "Nb", &Handle ); }
if (!NT_SUCCESS(Status)) { try_return(Status); }
HandleAllocated = TRUE; }
if ( Fcb->NodeTypeCode == NW_NTC_DCB ) {
//
// We can only rename files in the same directory
//
if ( RtlCompareUnicodeString( &NewPath, &OldPath, TRUE ) != 0 ) { try_return(Status = STATUS_NOT_SUPPORTED);
} else {
Status = ExchangeWithWait ( IrpContext, SynchronousResponseCallback, "SbJJ", NCP_DIR_FUNCTION, NCP_RENAME_DIRECTORY, Handle, &OldFileName, &NewFileName); }
} else {
//
// We have to close the handle associated with the Icb that
// is doing the rename. Close that handle or the rename will
// fail for sure.
//
if ( Icb->HasRemoteHandle ) {
Status2 = ExchangeWithWait( IrpContext, SynchronousResponseCallback, "F-r", NCP_CLOSE, Icb->Handle, sizeof( Icb->Handle ) );
Icb->HasRemoteHandle = FALSE;
#ifdef NWDBG
if ( NT_SUCCESS( Status2 ) ) { Status2 = ParseResponse( IrpContext, IrpContext->rsp, IrpContext->ResponseLength, "N" ); }
ASSERT(NT_SUCCESS(Status2)); #endif
}
//
// Do the file rename Ncp.
//
Status = ExchangeWithWait ( IrpContext, SynchronousResponseCallback, "FbbJbJ", NCP_RENAME_FILE, Handle, SEARCH_ALL_FILES, &OldFileName, Fcb->Vcb->Specific.Disk.Handle, &NewFullName); }
} else {
//
// We are going through the long name path. Ensure that the
// VCB supports long names.
//
if ( Icb->SuperType.Fcb->Vcb->Specific.Disk.LongNameSpace == LFN_NO_OS2_NAME_SPACE) { try_return( Status = STATUS_OBJECT_PATH_SYNTAX_BAD ); }
if (Icb->SuperType.Fcb->IcbCount != 1) { try_return( Status = STATUS_ACCESS_DENIED); }
//
// After a rename, the only operation allowed on the handle is an
// NtClose.
//
Icb->State = ICB_STATE_CLOSE_PENDING;
if ((irpSp->Parameters.SetFile.ReplaceIfExists ) && (TargetIcb->Exists)) {
// Delete the file
Status = ExchangeWithWait( IrpContext, SynchronousResponseCallback, "LbbW-DbC", NCP_LFN_DELETE_FILE, TargetFcb->Vcb->Specific.Disk.LongNameSpace, TargetFcb->Vcb->Specific.Disk.VolumeNumber, SEARCH_ALL_FILES, TargetFcb->Vcb->Specific.Disk.Handle, LFN_FLAG_SHORT_DIRECTORY, &TargetFcb->RelativeFileName );
#ifdef NWDBG
if ( NT_SUCCESS( Status ) ) { Status2 = ParseResponse( IrpContext, IrpContext->rsp, IrpContext->ResponseLength, "N" ); }
ASSERT(NT_SUCCESS(Status2)); #endif
}
if ( Fcb->NodeTypeCode == NW_NTC_DCB ) {
//
// We can only rename files in the same directory
//
if ( Fcb->Vcb != TargetFcb->Vcb ) { try_return(Status = STATUS_NOT_SUPPORTED);
} else {
Status = ExchangeWithWait ( IrpContext, SynchronousResponseCallback, "LbbWbDbbbDbbNN", NCP_LFN_RENAME_FILE, Fcb->Vcb->Specific.Disk.LongNameSpace, 0, // Rename flag
SEARCH_ALL_DIRECTORIES, Fcb->Vcb->Specific.Disk.VolumeNumber, Fcb->Vcb->Specific.Disk.Handle, LFN_FLAG_SHORT_DIRECTORY, OccurenceCount( &Fcb->RelativeFileName, OBJ_NAME_PATH_SEPARATOR ) + 1, Fcb->Vcb->Specific.Disk.VolumeNumber, Fcb->Vcb->Specific.Disk.Handle, LFN_FLAG_SHORT_DIRECTORY, OccurenceCount( &TargetFcb->RelativeFileName, OBJ_NAME_PATH_SEPARATOR ) + 1, &Fcb->RelativeFileName, &TargetFcb->RelativeFileName ); }
} else {
//
// We have to close the handle associated with the Icb that
// is doing the rename. Close that handle or the rename will
// fail for sure.
//
if ( Icb->HasRemoteHandle ) {
Status2 = ExchangeWithWait( IrpContext, SynchronousResponseCallback, "F-r", NCP_CLOSE, Icb->Handle, sizeof( Icb->Handle ) );
Icb->HasRemoteHandle = FALSE;
#ifdef NWDBG
if ( NT_SUCCESS( Status2 ) ) { Status2 = ParseResponse( IrpContext, IrpContext->rsp, IrpContext->ResponseLength, "N" ); }
ASSERT(NT_SUCCESS(Status2)); #endif
}
//
// Do the file rename Ncp.
//
Status = ExchangeWithWait ( IrpContext, SynchronousResponseCallback, "LbbWbDbbbDbbNN", NCP_LFN_RENAME_FILE, Fcb->Vcb->Specific.Disk.LongNameSpace, 0, // Rename flag
SEARCH_ALL_FILES, Fcb->Vcb->Specific.Disk.VolumeNumber, Fcb->Vcb->Specific.Disk.Handle, LFN_FLAG_SHORT_DIRECTORY, OccurenceCount( &Fcb->RelativeFileName, OBJ_NAME_PATH_SEPARATOR ) + 1, Fcb->Vcb->Specific.Disk.VolumeNumber, Fcb->Vcb->Specific.Disk.Handle, LFN_FLAG_SHORT_DIRECTORY, OccurenceCount( &TargetFcb->RelativeFileName, OBJ_NAME_PATH_SEPARATOR ) + 1, &Fcb->RelativeFileName, &TargetFcb->RelativeFileName ); } }
try_exit: NOTHING; } finally {
if (HandleAllocated) {
Status2 = ExchangeWithWait ( IrpContext, SynchronousResponseCallback, "Sb", // NCP Deallocate directory handle
NCP_DIR_FUNCTION, NCP_DEALLOCATE_DIR_HANDLE, Handle); #ifdef NWDBG
if ( NT_SUCCESS( Status2 ) ) { Status2 = ParseResponse( IrpContext, IrpContext->rsp, IrpContext->ResponseLength, "N" ); }
ASSERT(NT_SUCCESS(Status2)); #endif
}
NwReleaseFcb( Fcb->NonPagedFcb ); }
DebugTrace(-1, Dbg, "SetRenameInfo %08lx\n", Status );
//
// We're done with this request. Dequeue the IRP context from
// SCB and complete the request.
//
if ( Status != STATUS_PENDING ) { NwDequeueIrpContext( IrpContext, FALSE ); }
return Status; }
NTSTATUS NwSetPositionInfo ( IN PIRP_CONTEXT IrpContext, IN PICB Icb, IN PFILE_POSITION_INFORMATION Buffer ) /*++
Routine Description:
This routine sets position information for a file.
Arguments:
pIrpContext - A pointer to the IRP context information for the request in progress.
Icb - A pointer to the ICB of the file to set.
Buffer - The request buffer.
Return Value:
The status of the operation.
--*/ { PAGED_CODE();
ASSERT( Buffer->CurrentByteOffset.HighPart == 0 );
if ( Icb->FileObject ) { Icb->FileObject->CurrentByteOffset.QuadPart = Buffer->CurrentByteOffset.QuadPart; }
return( STATUS_SUCCESS ); }
NTSTATUS NwSetAllocationInfo ( IN PIRP_CONTEXT IrpContext, IN PICB Icb, IN PFILE_ALLOCATION_INFORMATION Buffer ) /*++
Routine Description:
This routine sets allocation information for a file.
Arguments:
pIrpContext - A pointer to the IRP context information for the request in progress.
Icb - A pointer to the ICB of the file to set.
Buffer - The request buffer.
Return Value:
The status of the operation.
--*/ { NTSTATUS Status; PIRP irp; PIO_STACK_LOCATION irpSp; PFCB fcb = (PFCB)Icb->SuperType.Fcb; PULONG pFileSize;
PAGED_CODE();
ASSERT( Buffer->AllocationSize.HighPart == 0);
if ( fcb->NodeTypeCode == NW_NTC_FCB ) {
pFileSize = &Icb->NpFcb->Header.FileSize.LowPart;
IrpContext->pNpScb = fcb->Scb->pNpScb;
if (BooleanFlagOn( fcb->Vcb->Flags, VCB_FLAG_PRINT_QUEUE ) ) { if (IsTerminalServer()) { // 2/10/97 cjc Fix problem for binary files not printing correctly
// if done via the COPY command. Works with NT RDR so
// changed this to behave same way.
return(STATUS_INVALID_PARAMETER); } else { return STATUS_SUCCESS; } }
} else if ( fcb->NodeTypeCode == NW_NTC_SCB ) {
pFileSize = &Icb->FileSize;
IrpContext->pNpScb = ((PSCB)fcb)->pNpScb;
} else {
DebugTrace(0, Dbg, "Not a file or a server\n", 0);
DebugTrace( 0, Dbg, "NwSetAllocationInfo -> %08lx\n", STATUS_INVALID_PARAMETER ); return STATUS_INVALID_PARAMETER; }
NwAppendToQueueAndWait( IrpContext );
if ( !Icb->HasRemoteHandle ) {
Status = STATUS_INVALID_PARAMETER;
} else if ( Buffer->AllocationSize.LowPart == *pFileSize ) {
Status = STATUS_SUCCESS;
} else {
irp = IrpContext->pOriginalIrp; irpSp = IoGetCurrentIrpStackLocation( irp );
#ifndef QFE_BUILD
if ( Buffer->AllocationSize.LowPart < *pFileSize ) {
//
// Before we actually truncate, check to see if the purge
// is going to fail.
//
if (!MmCanFileBeTruncated( irpSp->FileObject->SectionObjectPointer, &Buffer->AllocationSize )) {
return( STATUS_USER_MAPPED_FILE ); } } #endif
if ( fcb->NodeTypeCode == NW_NTC_FCB ) { AcquireFcbAndFlushCache( IrpContext, fcb->NonPagedFcb ); }
Status = ExchangeWithWait( IrpContext, SynchronousResponseCallback, "F-rd=", NCP_WRITE_FILE, &Icb->Handle, sizeof( Icb->Handle ), Buffer->AllocationSize.LowPart );
if ( NT_SUCCESS( Status ) ) { *pFileSize = Buffer->AllocationSize.LowPart; } }
NwDequeueIrpContext( IrpContext, FALSE );
return( Status ); }
NTSTATUS NwSetEndOfFileInfo ( IN PIRP_CONTEXT IrpContext, IN PICB Icb, IN PFILE_END_OF_FILE_INFORMATION Buffer ) /*++
Routine Description:
This routine sets end of file information for a file.
Arguments:
pIrpContext - A pointer to the IRP context information for the request in progress.
Icb - A pointer to the ICB of the file to set.
Buffer - The request buffer.
Return Value:
The status of the operation.
--*/ { NTSTATUS Status; PIRP irp; PIO_STACK_LOCATION irpSp; PFCB fcb = (PFCB)Icb->SuperType.Fcb; PULONG pFileSize;
PAGED_CODE();
ASSERT( Buffer->EndOfFile.HighPart == 0);
if ( fcb->NodeTypeCode == NW_NTC_FCB ) {
pFileSize = &Icb->NpFcb->Header.FileSize.LowPart;
IrpContext->pNpScb = fcb->Scb->pNpScb;
if (BooleanFlagOn( fcb->Vcb->Flags, VCB_FLAG_PRINT_QUEUE ) ) {
return STATUS_SUCCESS;
}
} else if ( fcb->NodeTypeCode == NW_NTC_SCB ) {
pFileSize = &Icb->FileSize;
IrpContext->pNpScb = ((PSCB)fcb)->pNpScb;
} else {
DebugTrace(0, Dbg, "Not a file or a server\n", 0);
DebugTrace( 0, Dbg, "NwSetAllocationInfo -> %08lx\n", STATUS_INVALID_PARAMETER ); return STATUS_INVALID_PARAMETER; }
NwAppendToQueueAndWait( IrpContext );
if ( !Icb->HasRemoteHandle ) {
Status = STATUS_INVALID_PARAMETER;
} else if ( Buffer->EndOfFile.LowPart == *pFileSize ) {
Status = STATUS_SUCCESS;
} else {
irp = IrpContext->pOriginalIrp; irpSp = IoGetCurrentIrpStackLocation( irp );
#ifndef QFE_BUILD
if ( Buffer->EndOfFile.LowPart < *pFileSize ) {
//
// Before we actually truncate, check to see if the purge
// is going to fail.
//
if (!MmCanFileBeTruncated( irpSp->FileObject->SectionObjectPointer, &Buffer->EndOfFile )) {
return( STATUS_USER_MAPPED_FILE ); } } #endif
if ( fcb->NodeTypeCode == NW_NTC_FCB ) { AcquireFcbAndFlushCache( IrpContext, fcb->NonPagedFcb ); }
Status = ExchangeWithWait( IrpContext, SynchronousResponseCallback, "F-rd=", NCP_WRITE_FILE, &Icb->Handle, sizeof( Icb->Handle ), Buffer->EndOfFile.LowPart );
if ( NT_SUCCESS( Status ) ) { *pFileSize = Buffer->EndOfFile.LowPart; } }
NwDequeueIrpContext( IrpContext, FALSE );
return( Status ); }
ULONG OccurenceCount ( IN PUNICODE_STRING String, IN WCHAR SearchChar ) /*++
Routine Description:
This routine counts the number of occurences of a search character in a string
Arguments:
String - The string to search
SearchChar - The character to search for.
Return Value:
The occurence count.
--*/ { PWCH currentChar; PWCH endOfString; ULONG count = 0;
PAGED_CODE();
currentChar = String->Buffer; endOfString = &String->Buffer[ String->Length / sizeof(WCHAR) ];
while ( currentChar < endOfString ) { if ( *currentChar == SearchChar ) { count++; } currentChar++; }
return( count ); }
|