|
|
/*++
Copyright (c) 1993 Microsoft Corporation
Module Name:
Create.c
Abstract:
This module implements the File Create routine for the NetWare redirector called by the dispatch driver.
Author:
Colin Watson [ColinW] 19-Dec-1992 Manny Weiser [MannyW] 15-Feb-1993
Revision History:
--*/
#include "Procs.h"
NTSTATUS NwCommonCreate ( IN PIRP_CONTEXT IrpContext );
IO_STATUS_BLOCK OpenRedirector( IN PIRP_CONTEXT IrpContext, ULONG DesiredAccess, ULONG ShareAccess, PFILE_OBJECT FileObject );
IO_STATUS_BLOCK CreateRemoteFile( IN PIRP_CONTEXT IrpContext, IN PUNICODE_STRING DriveName );
IO_STATUS_BLOCK ChangeDirectory( PIRP_CONTEXT IrpContext, PVCB Vcb, PICB Icb );
IO_STATUS_BLOCK CreateDir( PIRP_CONTEXT IrpContext, PVCB Vcb, PICB Icb );
NTSTATUS FileOrDirectoryExists( IN PIRP_CONTEXT IrpContext, IN PVCB Vcb, IN PICB Icb, PUNICODE_STRING Name, OUT PBOOLEAN IsAFile );
IO_STATUS_BLOCK OpenFile( IN PIRP_CONTEXT IrpContext, IN PVCB Vcb, IN PICB Icb, IN BYTE SearchFlags, IN BYTE ShareFlags );
IO_STATUS_BLOCK CreateNewFile( IN PIRP_CONTEXT IrpContext, IN PVCB Vcb, IN PICB Icb, IN BYTE SearchFlags, IN BYTE ShareFlags );
IO_STATUS_BLOCK CreateOrOverwriteFile( IN PIRP_CONTEXT IrpContext, IN PVCB Vcb, IN PICB Icb, IN BYTE CreateAttributes, IN BYTE OpenFlags, IN BOOLEAN CreateOperation );
IO_STATUS_BLOCK OpenRenameTarget( IN PIRP_CONTEXT IrpContext, IN PVCB Vcb, IN PDCB Dcb, IN PICB* Icb );
IO_STATUS_BLOCK CreatePrintJob( PIRP_CONTEXT IrpContext, PVCB Vcb, PICB Icb, PUNICODE_STRING DriveName );
VOID CloseFile( PIRP_CONTEXT pIrpContext, PICB pIcb );
BOOLEAN MmDisableModifiedWriteOfSection ( IN PSECTION_OBJECT_POINTERS SectionObjectPointer );
//
// The debug trace level
//
#define Dbg (DEBUG_TRACE_CREATE)
#ifdef ALLOC_PRAGMA
#pragma alloc_text( PAGE, NwFsdCreate )
#pragma alloc_text( PAGE, NwCommonCreate )
#pragma alloc_text( PAGE, ReadAttachEas )
#pragma alloc_text( PAGE, OpenRedirector )
#pragma alloc_text( PAGE, CreateRemoteFile )
#pragma alloc_text( PAGE, ChangeDirectory )
#pragma alloc_text( PAGE, CreateDir )
#pragma alloc_text( PAGE, FileOrDirectoryExists )
#pragma alloc_text( PAGE, OpenFile )
#pragma alloc_text( PAGE, CreateNewFile )
#pragma alloc_text( PAGE, CreateOrOverwriteFile )
#pragma alloc_text( PAGE, OpenRenameTarget )
#pragma alloc_text( PAGE, CreatePrintJob )
#pragma alloc_text( PAGE, CloseFile )
#endif
NTSTATUS NwFsdCreate ( IN PDEVICE_OBJECT DeviceObject, IN PIRP Irp )
/*++
Routine Description:
This routine implements the FSD part of the NtCreateFile and NtOpenFile API calls.
Arguments:
DeviceObject - Supplies the device object for the redirector.
Irp - Supplies the Irp being processed
Return Value:
NTSTATUS - The Fsd status for the Irp
--*/
{ NTSTATUS Status; PIRP_CONTEXT IrpContext = NULL; BOOLEAN TopLevel;
PAGED_CODE();
TimerStart(Dbg); DebugTrace(+1, Dbg, "NwFsdCreate\n", 0);
//
// Call the common create routine, with block allowed if the operation
// is synchronous.
//
FsRtlEnterFileSystem(); TopLevel = NwIsIrpTopLevel( Irp );
try {
IrpContext = AllocateIrpContext( Irp ); Status = NwCommonCreate( IrpContext );
} except( NwExceptionFilter( Irp, GetExceptionInformation() )) {
if ( IrpContext == 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( IrpContext, GetExceptionCode() ); } }
if ( IrpContext ) { NwDequeueIrpContext( IrpContext, FALSE ); NwCompleteRequest( IrpContext, Status ); }
if ( TopLevel ) { NwSetTopLevelIrp( NULL ); } FsRtlExitFileSystem();
//
// And return to our caller
//
DebugTrace(-1, Dbg, "NwFsdCreate -> %08lx\n", Status );
TimerStop(Dbg,"NwFsdCreate");
return Status;
UNREFERENCED_PARAMETER(DeviceObject); }
NTSTATUS NwCommonCreate ( IN PIRP_CONTEXT IrpContext )
/*++
Routine Description:
This is the common routine for creating/opening a file called by both the fsd and fsp threads.
Arguments:
IrpContext - Supplies the context information for the IRP to process
Return Value:
NTSTATUS - the return status for the operation
--*/
{ IO_STATUS_BLOCK Iosb; PIRP Irp; PIO_STACK_LOCATION IrpSp;
PFILE_OBJECT FileObject; ACCESS_MASK DesiredAccess; USHORT ShareAccess; ULONG Options; BOOLEAN CreateTreeConnection; BOOLEAN DeleteOnClose; BOOLEAN DeferredLogon; BOOLEAN DereferenceCodeSection = FALSE; BOOLEAN OpenedTreeHandle = FALSE;
BOOLEAN fNDSLookupFirst = FALSE; USHORT iBufferIndex = 0; DWORD dwSlashCount = 0;
UNICODE_STRING CreateFileName; UNICODE_STRING Drive; UNICODE_STRING Server; UNICODE_STRING Volume; UNICODE_STRING Path; UNICODE_STRING FileName; UNICODE_STRING UserName, Password; ULONG ShareType; WCHAR DriveLetter; DWORD dwExtendedCreate = FALSE;
PSCB Scb = NULL; PICB Icb; UNICODE_STRING DefaultServer; SECURITY_SUBJECT_CONTEXT SubjectContext;
PAGED_CODE();
//
// Get the current IRP stack location
//
Irp = IrpContext->pOriginalIrp;
IrpSp = IoGetCurrentIrpStackLocation( Irp );
//
// tommye - MS bug 30091 / MCS 262 - added some safety nets around those pointers
// containing pointers so we don't bugcheck in the debug code.
//
DebugTrace(+1, Dbg, "NwCommonCreate\n", 0 ); DebugTrace( 0, Dbg, "Irp = %08lx\n", Irp ); DebugTrace( 0, Dbg, "->Flags = %08lx\n", Irp->Flags ); DebugTrace( 0, Dbg, "->FileObject = %08lx\n", IrpSp->FileObject ); if (IrpSp->FileObject) { DebugTrace( 0, Dbg, " ->RelatedFileObject = %08lx\n", IrpSp->FileObject->RelatedFileObject ); DebugTrace( 0, Dbg, " ->FileName = \"%wZ\"\n", &IrpSp->FileObject->FileName ); } DebugTrace( 0, Dbg, "->AllocationSize.LowPart = %08lx\n", Irp->Overlay.AllocationSize.LowPart ); DebugTrace( 0, Dbg, "->AllocationSize.HighPart = %08lx\n", Irp->Overlay.AllocationSize.HighPart ); DebugTrace( 0, Dbg, "->SystemBuffer = %08lx\n", Irp->AssociatedIrp.SystemBuffer ); DebugTrace( 0, Dbg, "->IrpSp->Flags = %08lx\n", IrpSp->Flags ); if (IrpSp->Parameters.Create.SecurityContext) { DebugTrace( 0, Dbg, "->DesiredAccess = %08lx\n", IrpSp->Parameters.Create.SecurityContext->DesiredAccess ); } DebugTrace( 0, Dbg, "->Options = %08lx\n", IrpSp->Parameters.Create.Options ); DebugTrace( 0, Dbg, "->Disposition = %08lx\n", (IrpSp->Parameters.Create.Options >> 24) & 0x000000ff); DebugTrace( 0, Dbg, "->FileAttributes = %04x\n", IrpSp->Parameters.Create.FileAttributes ); DebugTrace( 0, Dbg, "->ShareAccess = %04x\n", IrpSp->Parameters.Create.ShareAccess ); DebugTrace( 0, Dbg, "->EaLength = %08lx\n", IrpSp->Parameters.Create.EaLength );
CreateFileName = IrpSp->FileObject->FileName; Options = IrpSp->Parameters.Create.Options; DesiredAccess = IrpSp->Parameters.Create.SecurityContext->DesiredAccess; ShareAccess = IrpSp->Parameters.Create.ShareAccess;
CreateTreeConnection = BooleanFlagOn( Options, FILE_CREATE_TREE_CONNECTION ); DeleteOnClose = BooleanFlagOn( Options, FILE_DELETE_ON_CLOSE );
DefaultServer.Buffer = NULL;
//
// Make sure the input large integer is valid
//
if (Irp->Overlay.AllocationSize.HighPart != 0) {
DebugTrace(-1, Dbg, "NwCommonCreate -> STATUS_INVALID_PARAMETER\n", 0); return STATUS_INVALID_PARAMETER; }
//
// Fail requests that don't have the proper impersonation level.
//
/* This test is overly restrictive and unnecessary
if ( IrpSp->Parameters.Create.SecurityContext ) {
if ( IrpSp->Parameters.Create.SecurityContext->SecurityQos ) {
if ( IrpSp->Parameters.Create.SecurityContext->SecurityQos->ImpersonationLevel < SecurityImpersonation ) {
DebugTrace(-1, Dbg, "NwCommonCreate -> Insufficient impersation level.\n", 0); return STATUS_ACCESS_DENIED; } } } */
Iosb.Status = STATUS_SUCCESS;
FileObject = IrpSp->FileObject; IrpContext->pNpScb = NULL;
IrpContext->Specific.Create.UserUid = GetUid(&IrpSp->Parameters.Create.SecurityContext->AccessState->SubjectSecurityContext);
try {
if ( IrpSp->FileObject->RelatedFileObject != NULL ) {
//
// If we open a handle then the DereferenceCodeSection flag
// will be set to false. The dereference will eventually
// happen when the file is closed.
//
NwReferenceUnlockableCodeSection(); DereferenceCodeSection = TRUE;
//
// Record the relative file name for this open.
//
IrpContext->Specific.Create.FullPathName = CreateFileName;
Iosb = CreateRemoteFile( IrpContext, NULL );
//
// If we succeeded, we want to keep the code section
// referenced because we have opened a handle.
//
if ( NT_SUCCESS( Iosb.Status ) ) { DereferenceCodeSection = FALSE; }
try_return( Iosb.Status ); }
Iosb.Status = CrackPath ( &CreateFileName, &Drive, &DriveLetter, &Server, &Volume, &Path, &FileName, NULL );
if ( !NT_SUCCESS(Iosb.Status)) { try_return(Iosb.Status); }
//
// Remember this good info.
//
IrpContext->Specific.Create.VolumeName = Volume; IrpContext->Specific.Create.PathName = Path; IrpContext->Specific.Create.DriveLetter = DriveLetter; IrpContext->Specific.Create.FileName = FileName; IrpContext->Specific.Create.FullPathName = CreateFileName;
RtlInitUnicodeString( &IrpContext->Specific.Create.UidConnectName, NULL );
//
// For now assume default username and password
//
ShareType = RESOURCETYPE_ANY; RtlInitUnicodeString( &UserName, NULL ); RtlInitUnicodeString( &Password, NULL );
if ((Server.Length == 0) && (CreateFileName.Length == 0)) {
//
// Opened the redirector itself
//
Iosb = OpenRedirector( IrpContext, DesiredAccess, ShareAccess, FileObject );
} else if ( Server.Length == Volume.Length - sizeof( WCHAR ) ) {
if (IpxHandle == 0 ) {
//
// We're not bound to the transport and the user is not
// opening the redirector to tell us to bind so return failed.
//
try_return( Iosb.Status = STATUS_REDIRECTOR_NOT_STARTED ); }
NwReferenceUnlockableCodeSection(); DereferenceCodeSection = TRUE;
//
// If the only requested access is FILE_LIST_DIRECTORY,
// defer the logon. This will allow all CreateScb to
// succeed with when the user or password is invalid, so
// that the user can see volumes, or enumerate servers
// on the server.
//
if ( (DesiredAccess & ~( FILE_LIST_DIRECTORY | SYNCHRONIZE ) ) == 0 ) { DeferredLogon = TRUE; } else { DeferredLogon = FALSE; }
//
// Server = "Server", Volume = "\Server"
//
if ( Server.Length == sizeof(WCHAR) && Server.Buffer[0] == L'*') {
//
// Attempt to open \\*, open a handle to the preferred
// server
//
PLOGON Logon;
NwAcquireExclusiveRcb( &NwRcb, TRUE );
Logon = FindUser( &IrpContext->Specific.Create.UserUid, FALSE); ASSERT( Logon != NULL );
//
// Capture the name to avoid holding Rcb or referencing
// the logon structure.
//
Iosb.Status = DuplicateUnicodeStringWithString ( &DefaultServer, &Logon->ServerName, PagedPool);
NwReleaseRcb( &NwRcb );
if (!NT_SUCCESS(Iosb.Status)) { try_return( Iosb.Status ); }
//
// If the user specified a preferred server and we managed
// to capture the name, try and connect to it.
//
if (DefaultServer.Length != 0) {
Iosb.Status = CreateScb( &Scb, IrpContext, &DefaultServer, NULL, NULL, NULL, DeferredLogon, FALSE );
} else {
//
// Record that we could not get to the server specified
// in the login structure and that we should attempt to
// use the nearest server.
//
Iosb.Status = STATUS_BAD_NETWORK_PATH; }
if ( !NT_SUCCESS(Iosb.Status)) {
PNONPAGED_SCB NpScb;
//
// First dequeue the IRP context, in case it was left
// on an SCB queue.
//
NwDequeueIrpContext( IrpContext, FALSE );
//
// Cannot get to the Preferred server so use any
// server we have a connection to.
//
NpScb = SelectConnection( NULL );
if (NpScb != NULL ) {
Scb = NpScb->pScb;
Iosb.Status = CreateScb( &Scb, IrpContext, &NpScb->ServerName, NULL, NULL, NULL, DeferredLogon, FALSE );
//
// Release the SCB reference we obtained from
// SelectConnection().
//
NwDereferenceScb( NpScb ); } }
if ( !NT_SUCCESS(Iosb.Status)) {
//
// First dequeue the IRP context, in case it was left
// on an SCB queue.
//
NwDequeueIrpContext( IrpContext, FALSE );
//
// Let CreateScb try and find a nearest server to talk
// to.
//
Iosb.Status = CreateScb( &Scb, IrpContext, NULL, NULL, NULL, NULL, DeferredLogon, FALSE ); }
if ( !NT_SUCCESS(Iosb.Status)) { try_return( Iosb.Status ); }
} else {
//
// On handle opens to a server or tree we support the concept
// of an open with supplemental credentials. In this case, we return
// a handle to the server or a dir server using the provided
// credentials regardless of whether or not there are existing
// connections to the resource. This is primarily for admin
// tools like OleDs.
//
ReadAttachEas( Irp, &UserName, &Password, &ShareType, &dwExtendedCreate );
if ( dwExtendedCreate ) {
ASSERT( UserName.Length > 0 );
IrpContext->Specific.Create.fExCredentialCreate = TRUE; IrpContext->Specific.Create.puCredentialName = &UserName;
//
// Reference the credentials before doing the create so
// we are guaranteed not to lose them. This call will
// create a credential shell if none exists. This keeps
// our reference counting consistent. We track the
// credentials pointer in the irp context specific data.
//
Iosb.Status = ExCreateReferenceCredentials( IrpContext, &Server );
if ( !NT_SUCCESS( Iosb.Status ) ) { try_return( Iosb.Status ); }
} if (PreferNDSBrowsing) {
//
// Attempt to open \\TREE
//
Iosb.Status = NdsCreateTreeScb( IrpContext, &Scb, // dest scb
&Server, // tree we want
&UserName, &Password, DeferredLogon, DeleteOnClose ); if ( NT_SUCCESS( Iosb.Status ) ) { OpenedTreeHandle = TRUE; } if ( ( Iosb.Status == STATUS_REMOTE_NOT_LISTENING ) || ( Iosb.Status == STATUS_BAD_NETWORK_PATH ) || ( Iosb.Status == STATUS_UNSUCCESSFUL ) ) { //
// If we couldn't find the server or something
// inexplicable occurred, attempt to open \\server
//
Iosb.Status = CreateScb( &Scb, IrpContext, &Server, NULL, &UserName, &Password, DeferredLogon, DeleteOnClose ); } }else{ //
// Attempt to open \\server
//
Iosb.Status = CreateScb( &Scb, IrpContext, &Server, NULL, &UserName, &Password, DeferredLogon, DeleteOnClose );
if ( ( Iosb.Status == STATUS_REMOTE_NOT_LISTENING ) || ( Iosb.Status == STATUS_BAD_NETWORK_PATH ) || ( Iosb.Status == STATUS_UNSUCCESSFUL ) ) {
//
// If we couldn't find the server or something
// inexplicable occurred, attempt to open \\tree.
//
Iosb.Status = NdsCreateTreeScb( IrpContext, &Scb, // dest scb
&Server, // tree we want
&UserName, &Password, DeferredLogon, DeleteOnClose ); if ( NT_SUCCESS( Iosb.Status ) ) { OpenedTreeHandle = TRUE; } } }
// if( IsTerminalServer() ) clause below has been shifted down as we are more
// likely to be opening a tree or server than a pserver.
// so we need to check there first.
if (IsTerminalServer()) { /*
* This is an attempt to get GUEST to work for printman. * I.E. If you have no connection, try the guest * connection. */ if ( ( !NT_SUCCESS(Iosb.Status) ) && ( Iosb.Status == STATUS_NO_SUCH_USER ) && ( !CreateTreeConnection ) && ( !DeferredLogon ) ) {
DebugTrace( -1, Dbg, " Attempting default GUEST logon for %wZ\n", &Server );
Iosb.Status = CreateScb( &Scb, IrpContext, &Server, NULL, &Guest.UserName, &Guest.PassWord, DeferredLogon, DeleteOnClose ); } }
if ( !NT_SUCCESS( Iosb.Status ) ) {
//
// If we failed to get the bindery server for
// some legitimate reason, bail out now.
//
try_return( Iosb.Status ); }
//
// We must have a connection at this point. We don't tree
// connect the dir server since it's virtual.
//
if ( !OpenedTreeHandle && CreateTreeConnection && !DeleteOnClose ) { TreeConnectScb( Scb ); }
}
//
// Now create the ICB.
//
ASSERT( Iosb.Status == STATUS_SUCCESS ); ASSERT( Scb != NULL );
Icb = NwCreateIcb( NW_NTC_ICB_SCB, Scb ); Icb->FileObject = FileObject; NwSetFileObject( FileObject, NULL, Icb );
//
// Indicate that the SCB was opened.
//
Icb->State = ICB_STATE_OPENED;
//
// Is this a tree handle?
//
Icb->IsTreeHandle = OpenedTreeHandle;
//
// If this was an extended create, associate this handle
// with its extended credentials so that we can cleanup
// when all the handles are closed.
//
if ( IrpContext->Specific.Create.fExCredentialCreate ) {
ASSERT( IrpContext->Specific.Create.pExCredentials != NULL ); Icb->pContext = IrpContext->Specific.Create.pExCredentials; Icb->IsExCredentialHandle = TRUE;
}
} else {
NwReferenceUnlockableCodeSection(); DereferenceCodeSection = TRUE;
DeferredLogon = FALSE;
if ( CreateTreeConnection ) {
//
// We ignore the extended create attribute here because
// we DO NOT support extended credential creates to random
// files and directories!
//
ReadAttachEas( Irp, &UserName, &Password, &ShareType, NULL );
if ( DeleteOnClose ) {
//
// Opening a directory to delete a volume. Do not
// force logon.
//
DeferredLogon = TRUE; } } IrpContext->Specific.Create.ShareType = ShareType; IrpContext->Specific.Create.NdsCreate = FALSE; //
// Check to see if this is an NDS object, if so set the flag to check NDS first.
// The only way a DOT can be in the Volume name is if it is an NDS Object,
// between the third and fourth slashes.
//
fNDSLookupFirst = FALSE;
for (iBufferIndex=0; iBufferIndex < (USHORT)(Volume.Length/sizeof(WCHAR)); iBufferIndex++ ) { if (Volume.Buffer[iBufferIndex] == L'\\') dwSlashCount++; if (dwSlashCount > 3) { fNDSLookupFirst = FALSE; break; } if (Volume.Buffer[iBufferIndex] == L'.') { fNDSLookupFirst = TRUE; break; } }
if (fNDSLookupFirst) {
IrpContext->Specific.Create.NdsCreate = TRUE; IrpContext->Specific.Create.NeedNdsData = TRUE; Iosb.Status = NdsCreateTreeScb( IrpContext, &Scb, &Server, &UserName, &Password, DeferredLogon, DeleteOnClose );
if ( Iosb.Status == STATUS_REMOTE_NOT_LISTENING || Iosb.Status == STATUS_BAD_NETWORK_PATH || Iosb.Status == STATUS_UNSUCCESSFUL ) { //
// Not found, do a bindery lookup
//
IrpContext->Specific.Create.NdsCreate = FALSE; IrpContext->Specific.Create.NeedNdsData = FALSE; Iosb.Status = CreateScb( &Scb, IrpContext, &Server, NULL, &UserName, &Password, DeferredLogon, DeleteOnClose ); } }else { //
// Object appears to be bindery, check there first.
//
Iosb.Status = CreateScb( &Scb, IrpContext, &Server, NULL, &UserName, &Password, DeferredLogon, DeleteOnClose );
if ( Iosb.Status == STATUS_REMOTE_NOT_LISTENING || Iosb.Status == STATUS_BAD_NETWORK_PATH || Iosb.Status == STATUS_UNSUCCESSFUL ) {
//
// If we couldn't find the server or something
// inexplicable occurred, attempt to open \\tree.
//
IrpContext->Specific.Create.NdsCreate = TRUE; IrpContext->Specific.Create.NeedNdsData = TRUE;
Iosb.Status = NdsCreateTreeScb( IrpContext, &Scb, &Server, &UserName, &Password, DeferredLogon, DeleteOnClose ); } }
//
// If we have success, then there's a volume to connect.
//
if ( NT_SUCCESS( Iosb.Status ) ) {
NTSTATUS CreateScbStatus;
ASSERT( Scb != NULL );
//
// Remember the status from create SCB, since it might
// be an interesting warning.
//
CreateScbStatus = Iosb.Status;
//
// We catch this exception in case we have to retry the
// create on the NDS path. This is horrable, as does the
// exception structure in this code right now, but it's
// legacy and now is not the time to change it.
//
try {
Iosb = CreateRemoteFile( IrpContext, &Drive );
} except ( EXCEPTION_EXECUTE_HANDLER ) {
Iosb.Status = GetExceptionCode(); }
//
// If this is a server whose name is the same as the tree
// that it is a member of, and the create was marked as
// non-nds and it failed, retry an nds create.
//
if ( ( !NT_SUCCESS( Iosb.Status) ) && ( !(IrpContext->Specific.Create.NdsCreate) ) && ( RtlEqualUnicodeString( &(Scb->pNpScb->ServerName), &(Scb->NdsTreeName), TRUE ) ) ) {
IrpContext->Specific.Create.NdsCreate = TRUE; IrpContext->Specific.Create.NeedNdsData = TRUE;
Iosb = CreateRemoteFile( IrpContext, &Drive );
//
// If this fails, it will raise status before setting IOSB
// and we'll return the status from the original create,
// which is the more interesting one.
//
}
//
// If we successfully open the remote file, return the
// CreateScb status instead.
//
if ( NT_SUCCESS( Iosb.Status ) ) { Iosb.Status = CreateScbStatus; }
} }
//
// If we succeeded, we want to keep the code section
// referenced because we have opened a handle.
//
if ( NT_SUCCESS( Iosb.Status ) ) { DereferenceCodeSection = FALSE; }
try_exit: NOTHING; } finally {
//
// Track the Scb in the IrpContext, not in the local Scb
// variable since we may have been routed to another server
// in process.
//
if (( Scb != NULL ) && ( IrpContext->pNpScb != NULL )) { NwDereferenceScb( IrpContext->pNpScb ); }
if ( DefaultServer.Buffer != NULL ) { FREE_POOL( DefaultServer.Buffer ); }
if ( ( IrpContext->Specific.Create.fExCredentialCreate ) && ( IrpContext->Specific.Create.pExCredentials ) && ( !NT_SUCCESS( Iosb.Status ) ) ) {
ExCreateDereferenceCredentials( IrpContext, IrpContext->Specific.Create.pExCredentials ); }
DebugTrace(-1, Dbg, "NwCommonCreate -> %08lx\n", Iosb.Status);
if ( DereferenceCodeSection ) { NwDereferenceUnlockableCodeSection (); }
}
//
// Map a timeout error to server not found, so that MPR will
// try to connect on the next network provider instead of giving up,
// which is wrong wrong wrong.
//
if ( Iosb.Status == STATUS_REMOTE_NOT_LISTENING ) { Iosb.Status = STATUS_BAD_NETWORK_PATH; }
//
// Map an unbound transport error to server not found, so that MPR
// will try to connect on the next provider.
//
if ( Iosb.Status == STATUS_NETWORK_UNREACHABLE ) { Iosb.Status = STATUS_BAD_NETWORK_PATH; }
return Iosb.Status; }
NTSTATUS ReadAttachEas( IN PIRP Irp, OUT PUNICODE_STRING UserName, OUT PUNICODE_STRING Password, OUT PULONG ShareType, OUT PDWORD CredentialExtension )
/*++
Routine Description:
This routine processes the EAs provided when the caller attempts to attach to a remote server.
Note: This routine does not create additional storage for the names. It is the callers responsibility to save them if required.
Arguments:
Irp - Supplies all the information
UserName - Returns the value of the User name EA
Password - Returns the value of the password EA
ShareType - Returns the value of the share type EA
CredentialExtension - Returns whether or not this create should use the provided credentials for an credential extended connection. This is primarily for OleDs accessing the ds in multiple security contexts.
Return Value:
NTSTATUS - Status of operation
--*/ {
PIO_STACK_LOCATION IrpSp = IoGetCurrentIrpStackLocation(Irp); PFILE_FULL_EA_INFORMATION EaBuffer = Irp->AssociatedIrp.SystemBuffer;
PAGED_CODE();
RtlInitUnicodeString( UserName, NULL ); RtlInitUnicodeString( Password, NULL ); *ShareType = RESOURCETYPE_ANY; if ( CredentialExtension ) { *CredentialExtension = FALSE; }
DebugTrace(+1, Dbg, "ReadAttachEas....\n", 0);
if ( EaBuffer != NULL) {
while (TRUE) { ULONG EaNameLength = EaBuffer->EaNameLength;
if (strcmp(EaBuffer->EaName, EA_NAME_USERNAME) == 0) {
UserName->Length = EaBuffer->EaValueLength; UserName->MaximumLength = EaBuffer->EaValueLength; UserName->Buffer = (PWSTR)(EaBuffer->EaName+EaNameLength+1);
} else if (strcmp(EaBuffer->EaName, EA_NAME_PASSWORD) == 0) {
Password->Length = EaBuffer->EaValueLength; Password->MaximumLength = EaBuffer->EaValueLength; Password->Buffer = (PWSTR)(EaBuffer->EaName+EaNameLength+1);
} else if ((strcmp(EaBuffer->EaName, EA_NAME_TYPE) == 0) && (EaBuffer->EaValueLength >= sizeof(ULONG))) {
*ShareType = *(ULONG UNALIGNED *)(EaBuffer->EaName+EaNameLength+1);
} else if (strcmp(EaBuffer->EaName, EA_NAME_CREDENTIAL_EX) == 0) {
if ( CredentialExtension ) { *CredentialExtension = TRUE; DebugTrace(0, Dbg, "ReadAttachEas signals a credential extension.\n", 0 ); }
} else { DebugTrace(0, Dbg, "ReadAttachEas Unknown EA -> %s\n", EaBuffer->EaName); }
if (EaBuffer->NextEntryOffset == 0) { break; } else { EaBuffer = (PFILE_FULL_EA_INFORMATION) ((PCHAR) EaBuffer+EaBuffer->NextEntryOffset); } } }
DebugTrace(-1, Dbg, "ReadAttachEas -> %08lx\n", STATUS_SUCCESS);
return STATUS_SUCCESS;
}
IO_STATUS_BLOCK OpenRedirector( IN PIRP_CONTEXT IrpContext, ULONG DesiredAccess, ULONG ShareAccess, PFILE_OBJECT FileObject )
/*++
Routine Description:
This routines opens a handle to the redirector device.
Arguments:
IrpContext - Supplies all the information
DesiredAccess - The requested access to the redirector.
ShareAccess - The requested share access to the redirector.
FileObject - A pointer to the caller file object.
Return Value:
IO_STATUS_BLOCK - Status of operation
--*/
{ IO_STATUS_BLOCK iosb;
PAGED_CODE();
//
// Note that the object manager will only allow an administrator
// to open the redir itself. This is good.
//
DebugTrace(+1, Dbg, "NwOpenRedirector\n", 0);
NwAcquireExclusiveRcb( &NwRcb, TRUE );
try {
//
// Set the new share access
//
if (!NT_SUCCESS(iosb.Status = IoCheckShareAccess( DesiredAccess, ShareAccess, FileObject, &NwRcb.ShareAccess, TRUE ))) {
DebugTrace(0, Dbg, "bad share access\n", 0);
try_return( NOTHING ); }
NwSetFileObject( FileObject, NULL, &NwRcb ); ++NwRcb.OpenCount;
//
// Set the return status.
//
iosb.Status = STATUS_SUCCESS; iosb.Information = FILE_OPENED;
try_exit: NOTHING; } finally {
NwReleaseRcb( &NwRcb ); DebugTrace(-1, Dbg, "NwOpenRedirector -> Iosb.Status = %08lx\n", iosb.Status);
}
//
// Return to the caller.
//
return iosb; }
IO_STATUS_BLOCK CreateRemoteFile( IN PIRP_CONTEXT IrpContext, IN PUNICODE_STRING DriveName ) /*++
Routine Description:
This routines opens a remote file or directory.
Arguments:
IrpContext - Supplies all the information
DriveName - The drive name. One of three forms X:, LPTx, or NULL.
Return Value:
IO_STATUS_BLOCK - Status of operation
--*/ { IO_STATUS_BLOCK Iosb; PIRP Irp; PIO_STACK_LOCATION IrpSp;
ULONG DesiredAccess; ULONG ShareAccess; PFILE_OBJECT FileObject;
UNICODE_STRING FileName; PFILE_OBJECT RelatedFileObject; ULONG Options; ULONG FileAttributes;
BOOLEAN CreateDirectory; BOOLEAN OpenDirectory; BOOLEAN DirectoryFile; BOOLEAN NonDirectoryFile; BOOLEAN DeleteOnClose; BOOLEAN OpenTargetDirectory; ULONG AllocationSize;
// Unhandled open features.
// PFILE_FULL_EA_INFORMATION EaBuffer;
// ULONG EaLength;
// BOOLEAN SequentialOnly;
// BOOLEAN NoIntermediateBuffering;
// BOOLEAN IsPagingFile;
// BOOLEAN NoEaKnowledge;
ULONG CreateDisposition;
PFCB Fcb = NULL; PICB Icb = NULL; PDCB Dcb; PVCB Vcb = NULL; PSCB Scb;
BOOLEAN IsAFile; BOOLEAN MayBeADirectory = FALSE; BOOLEAN OwnOpenLock = FALSE; BOOLEAN SetShareAccess = FALSE;
BYTE SearchFlags; BYTE ShareFlags;
BOOLEAN CreateTreeConnection = FALSE; PUNICODE_STRING VolumeName;
NTSTATUS Status; UNICODE_STRING NdsConnectName; WCHAR ConnectBuffer[MAX_NDS_NAME_CHARS]; BOOLEAN MadeUidNdsName = FALSE;
PAGED_CODE();
Irp = IrpContext->pOriginalIrp; IrpSp = IoGetCurrentIrpStackLocation( Irp ); DesiredAccess = IrpSp->Parameters.Create.SecurityContext->DesiredAccess; ShareAccess = IrpSp->Parameters.Create.ShareAccess; FileObject = IrpSp->FileObject; OpenTargetDirectory = BooleanFlagOn( IrpSp->Flags, SL_OPEN_TARGET_DIRECTORY );
//
// It is ok to attempt a reconnect if this request fails with a
// connection error.
//
SetFlag( IrpContext->Flags, IRP_FLAG_RECONNECTABLE );
try {
//
// Reference our input parameters to make things easier
//
RelatedFileObject = FileObject->RelatedFileObject;
//
// We actually want the parsed file name.
// FileName = FileObject->FileName;
//
FileName = IrpContext->Specific.Create.FullPathName; Options = IrpSp->Parameters.Create.Options; FileAttributes = IrpSp->Parameters.Create.FileAttributes; AllocationSize = Irp->Overlay.AllocationSize.LowPart;
//
// Short circuit an attempt to open a wildcard name.
//
if ( FsRtlDoesNameContainWildCards( &FileName ) ) { try_return( Iosb.Status = STATUS_OBJECT_NAME_INVALID ); }
// Decipher Option flags and values
//
DirectoryFile = BooleanFlagOn( Options, FILE_DIRECTORY_FILE ); NonDirectoryFile = BooleanFlagOn( Options, FILE_NON_DIRECTORY_FILE ); DeleteOnClose = BooleanFlagOn( Options, FILE_DELETE_ON_CLOSE );
//
// Things we currently ignore, because netware servers don't support it.
//
// SequentialOnly = BooleanFlagOn( Options, FILE_SEQUENTIAL_ONLY );
// NoIntermediateBuffering = BooleanFlagOn( Options, FILE_NO_INTERMEDIATE_BUFFERING );
// NoEaKnowledge = BooleanFlagOn( Options, FILE_NO_EA_KNOWLEDGE );
// EaBuffer = Irp->AssociatedIrp.SystemBuffer;
// EaLength = IrpSp->Parameters.Create.EaLength;
// IsPagingFile = BooleanFlagOn( IrpSp->Flags, SL_OPEN_PAGING_FILE );
if ( BooleanFlagOn( Options, FILE_CREATE_TREE_CONNECTION ) ) { CreateDisposition = FILE_OPEN; } else { CreateDisposition = (Options >> 24) & 0x000000ff; }
CreateDirectory = (BOOLEAN)(DirectoryFile && ((CreateDisposition == FILE_CREATE) || (CreateDisposition == FILE_OPEN_IF)));
OpenDirectory = (BOOLEAN)(DirectoryFile && ((CreateDisposition == FILE_OPEN) || (CreateDisposition == FILE_OPEN_IF)));
Dcb = NULL; if ( RelatedFileObject != NULL ) {
PNONPAGED_DCB NonPagedDcb;
NonPagedDcb = RelatedFileObject->FsContext; if ( NonPagedDcb ) { Dcb = NonPagedDcb->Fcb; }
//
// If there is a related file object then this is a relative open
// and it better be a DCB.
//
if ( !Dcb || (NodeType( Dcb ) != NW_NTC_DCB) ) {
DebugTrace(0, Dbg, "Bad file name\n", 0); Iosb.Status = STATUS_OBJECT_NAME_INVALID; try_return( Iosb ); }
//
// Obtain SCB pointers.
//
IrpContext->pScb = Dcb->Scb; IrpContext->pNpScb = Dcb->Scb->pNpScb; }
//
// We are about ready to send a packet. Append this IRP context
// the SCB workqueue, and wait until it gets to the front.
//
NwAppendToQueueAndWait( IrpContext ); ASSERT( IrpContext->pNpScb->Requests.Flink == &IrpContext->NextRequest );
//
// Acquire the Global FCB resource to ensure that one thread
// can't access the half created FCB of another thread.
//
NwAcquireOpenLock( ); OwnOpenLock = TRUE;
//
// Find the volume for this file.
//
CreateTreeConnection = BooleanFlagOn( Options, FILE_CREATE_TREE_CONNECTION );
if ( CreateTreeConnection ) { VolumeName = &IrpContext->Specific.Create.FullPathName; } else { VolumeName = &IrpContext->Specific.Create.VolumeName; }
if ( Dcb == NULL ) {
RetryFindVcb:
Vcb = NwFindVcb( IrpContext, VolumeName, IrpContext->Specific.Create.ShareType, IrpContext->Specific.Create.DriveLetter, CreateTreeConnection, ( BOOLEAN )( CreateTreeConnection && DeleteOnClose ) );
if ( Vcb == NULL ) {
//
// If this create failed because we need nds data, get
// the data from the ds and resubmit the request.
//
if ( IrpContext->Specific.Create.NdsCreate && IrpContext->Specific.Create.NeedNdsData ) {
//
// Release the open resource so we can move around.
//
NwReleaseOpenLock( ); OwnOpenLock = FALSE;
//
// Take the volume name and build the server/share
// connect name.
//
NdsConnectName.Buffer = ConnectBuffer; NdsConnectName.MaximumLength = sizeof( ConnectBuffer ); NdsConnectName.Length = 0;
//
// Get the ds information. We may jump servers here.
//
Status = NdsMapObjectToServerShare( IrpContext, &Scb, &NdsConnectName, CreateTreeConnection, &(IrpContext->Specific.Create.dwNdsOid) );
if( !NT_SUCCESS( Status ) ) { ExRaiseStatus( Status ); }
//
// Make sure we are on the scb queue after all the
// possible server jumping.
//
NwAppendToQueueAndWait( IrpContext );
NwAcquireOpenLock( ); OwnOpenLock = TRUE;
//
// Prepend the Uid to the server/share name.
//
MergeStrings( &IrpContext->Specific.Create.UidConnectName, &Scb->UnicodeUid, &NdsConnectName, PagedPool );
MadeUidNdsName = TRUE;
//
// We have the data, so re-do the connect.
//
IrpContext->Specific.Create.NeedNdsData = FALSE; goto RetryFindVcb;
} else {
//
// If this was an open to delete a tree connect, and we failed
// to find the VCB, simply return the error.
//
Iosb.Status = STATUS_BAD_NETWORK_PATH; try_return ( Iosb );
}
}
} else {
Vcb = Dcb->Vcb; NwReferenceVcb( Vcb );
}
ASSERT( Vcb->Scb == IrpContext->pScb );
//
// If this is the target name for a rename then we want to find the
// DCB for the parent directory.
//
if (OpenTargetDirectory) {
Iosb = OpenRenameTarget(IrpContext, Vcb, Dcb, &Icb ); if (Icb != NULL) { Fcb = Icb->SuperType.Fcb; } try_return ( Iosb );
}
//
// Find the FCB for this file. If the FCB exists, we get a
// referenced pointer. Otherwise a new FCB is created.
//
Fcb = NwFindFcb( IrpContext->pScb, Vcb, &FileName, Dcb ); // in rare cases, NwFindFcb might return NULL instead of throwing an exception
// Raid # 432500
if (Fcb == NULL) { DebugTrace(0, Dbg, "NwFindFcb returned NULL in CreateRemoteFile\n", 0); Iosb.Status = STATUS_INVALID_PARAMETER; try_return( Iosb ); }
//
// Check the share access for this file. The share access
// is updated if access is granted.
//
if (!IsTerminalServer() || !FlagOn( Vcb->Flags, VCB_FLAG_PRINT_QUEUE )) { if ( Fcb->IcbCount > 0 ) { NwAcquireSharedFcb( Fcb->NonPagedFcb, TRUE );
Iosb.Status = IoCheckShareAccess( DesiredAccess, ShareAccess, FileObject, &Fcb->ShareAccess, TRUE );
NwReleaseFcb( Fcb->NonPagedFcb );
if ( !NT_SUCCESS( Iosb.Status ) ) { try_return( Iosb ); }
} else {
NwAcquireExclusiveFcb( Fcb->NonPagedFcb, TRUE );
IoSetShareAccess( DesiredAccess, ShareAccess, FileObject, &Fcb->ShareAccess );
NwReleaseFcb( Fcb->NonPagedFcb ); }
SetShareAccess = TRUE; } //
// Now create the ICB.
//
Icb = NwCreateIcb( NW_NTC_ICB, Fcb ); Icb->FileObject = FileObject; NwSetFileObject( FileObject, Fcb->NonPagedFcb, Icb );
#ifndef QFE_BUILD
//
// Supply a resource for the modified page write to grab when
// writing mem mapped files. We do this because it is imposed
// on us by the system, we do not require the resource for any
// real serialization.
//
// This flag should not be zeroed (nealch: May 6, 2002)
//Fcb->NonPagedFcb->Header.Flags = 0;
Fcb->NonPagedFcb->Header.Resource = NULL;
#endif
#ifdef NWFASTIO
//
// Initialize private cache map so that the i/o system will call
// our fast path.
//
FileObject->PrivateCacheMap = (PVOID)1; #endif
IrpContext->Icb = Icb;
//
// Allocate an 8 bit PID for this ICB. Use different thread so
// each Wow program gets its own id. This is because if the same id
// has locks using two handles and closes just one of them the locks
// on that handle are not discarded.
//
Iosb.Status = NwMapPid(IrpContext->pNpScb, (ULONG_PTR)PsGetCurrentThread(), &Icb->Pid );
if ( !NT_SUCCESS( Iosb.Status ) ) { try_return( Iosb.Status ); }
//
// Try to figure out what it is we're expected to open.
//
Iosb.Status = STATUS_SUCCESS;
if ( FlagOn( Vcb->Flags, VCB_FLAG_PRINT_QUEUE ) ) {
//
// Opening a print queue job.
//
Iosb = CreatePrintJob( IrpContext, Vcb, Icb, DriveName );
} else if ( DirectoryFile || ( Fcb->State == FCB_STATE_OPENED && Fcb->NodeTypeCode == NW_NTC_DCB ) ) {
//
// Opening a directory.
//
MayBeADirectory = TRUE;
switch ( CreateDisposition ) {
case FILE_OPEN: Iosb = ChangeDirectory( IrpContext, Vcb, Icb ); break;
case FILE_CREATE: Iosb = CreateDir( IrpContext, Vcb, Icb ); break;
case FILE_OPEN_IF: Iosb.Status = FileOrDirectoryExists( IrpContext, Vcb, Icb, &Icb->SuperType.Fcb->RelativeFileName, &IsAFile );
//
// If the opener specified a directory, fail this request
// if the object is a file.
//
if ( NT_SUCCESS( Iosb.Status ) && IsAFile ) { Iosb.Status = STATUS_OBJECT_PATH_NOT_FOUND; } else if ( !NT_SUCCESS( Iosb.Status )) { Iosb = CreateDir( IrpContext, Vcb, Icb ); } break;
case FILE_SUPERSEDE: case FILE_OVERWRITE: case FILE_OVERWRITE_IF: Iosb.Status = STATUS_INVALID_PARAMETER; break;
default: KeBugCheck( RDR_FILE_SYSTEM );
}
} else {
SearchFlags = NtAttributesToNwAttributes( FileAttributes ); ShareFlags = NtToNwShareFlags( DesiredAccess, ShareAccess );
IsAFile = NonDirectoryFile || (Fcb->State == FCB_STATE_OPENED && Fcb->NodeTypeCode == NW_NTC_FCB ); //
// Assume we are opening a file. If that fails, and it makes
// sense try to open a directory.
//
switch ( CreateDisposition ) {
case FILE_OPEN:
//
// If the disposition is FILE_OPEN try to avoid an unneeded
// open, for some desired access types.
//
switch ( DesiredAccess & ~SYNCHRONIZE ) {
case FILE_WRITE_ATTRIBUTES: case FILE_READ_ATTRIBUTES: case DELETE:
Iosb.Status = FileOrDirectoryExists( IrpContext, Vcb, Icb, &Icb->SuperType.Fcb->RelativeFileName, &IsAFile );
if ( !IsAFile) { MayBeADirectory = TRUE; }
//
// Fail open of read only file for delete access,
// since the netware server won't fail the delete.
//
if ( NT_SUCCESS( Iosb.Status ) && CreateDisposition == DELETE && FlagOn( Icb->NpFcb->Attributes, NW_ATTRIBUTE_READ_ONLY ) ) {
Iosb.Status = STATUS_ACCESS_DENIED; }
if ( ( Iosb.Status == STATUS_OBJECT_NAME_NOT_FOUND ) && ( (DesiredAccess & ~SYNCHRONIZE) == DELETE ) ) { //
// we may not have scan rights. fake the return as OK.
// NW allows the delete without scan rights.
//
Iosb.Status = STATUS_SUCCESS; }
break;
default:
Iosb = OpenFile( IrpContext, Vcb, Icb, SearchFlags, ShareFlags );
if ( ( Iosb.Status == STATUS_OBJECT_NAME_NOT_FOUND || Iosb.Status == STATUS_FILE_IS_A_DIRECTORY ) && !IsAFile) {
//
// Opener didn't specify file or directory, and open
// file failed. So try open directory.
//
Iosb = ChangeDirectory( IrpContext, Vcb, Icb ); MayBeADirectory = TRUE;
} else if ( (Iosb.Status == STATUS_SHARING_VIOLATION) && ((ShareFlags == (NW_OPEN_FOR_READ | NW_DENY_WRITE)) || (ShareFlags == (NW_OPEN_FOR_READ)))) {
//
// if the file was already open exclusive (eg. GENERIC_EXECUTE)
// then a debugger opening it again for read will fail with
// sharing violation. In this case, we will try open exclusive
// again to see if that passes.
//
ShareFlags |= NW_OPEN_EXCLUSIVE ; ShareFlags &= ~(NW_DENY_WRITE | NW_DENY_READ); Iosb = OpenFile( IrpContext, Vcb, Icb, SearchFlags, ShareFlags ); }
break;
}
break;
case FILE_CREATE: Iosb = CreateNewFile( IrpContext, Vcb, Icb, SearchFlags, ShareFlags ); break;
case FILE_OPEN_IF: Iosb.Status = FileOrDirectoryExists( IrpContext, Vcb, Icb, &Icb->SuperType.Fcb->RelativeFileName, &IsAFile );
if ( NT_SUCCESS( Iosb.Status ) ) { Iosb = OpenFile( IrpContext, Vcb, Icb, SearchFlags, ShareFlags ); } else { Iosb = CreateNewFile( IrpContext, Vcb, Icb, SearchFlags, ShareFlags ); }
if ( !NT_SUCCESS( Iosb.Status ) && !IsAFile) {
//
// Opener didn't specify file or directory, and open
// file and create new file both failed. So try open
// or create directory.
//
MayBeADirectory = TRUE; Iosb.Status = FileOrDirectoryExists( IrpContext, Vcb, Icb, &Icb->SuperType.Fcb->RelativeFileName, &IsAFile);
if ( NT_SUCCESS( Iosb.Status ) ) { Iosb.Information = FILE_OPENED; } else { Iosb = CreateDir( IrpContext, Vcb, Icb ); } }
break;
//
// None of the below make sense for directories so if the
// file operation fails, just return the failure status
// to the user.
//
case FILE_SUPERSEDE: case FILE_OVERWRITE_IF:
//
// Actually, if Overwrite is chosen, we are supposed to
// get the attributes for a file and OR them with the
// new attributes.
//
Iosb = CreateOrOverwriteFile( IrpContext, Vcb, Icb, SearchFlags, ShareFlags, FALSE ); break;
case FILE_OVERWRITE: Iosb.Status = FileOrDirectoryExists( IrpContext, Vcb, Icb, &Icb->SuperType.Fcb->RelativeFileName, &IsAFile );
if ( NT_SUCCESS( Iosb.Status ) ) { Iosb = CreateOrOverwriteFile( IrpContext, Vcb, Icb, SearchFlags, ShareFlags, FALSE ); }
break;
default: KeBugCheck( RDR_FILE_SYSTEM ); }
}
try_exit: NOTHING;
} finally {
if ( Vcb != NULL ) { NwDereferenceVcb( Vcb, IrpContext, FALSE ); }
if ( MadeUidNdsName ) { FREE_POOL( IrpContext->Specific.Create.UidConnectName.Buffer ); }
if ( AbnormalTermination() || !NT_SUCCESS( Iosb.Status ) ) {
//
// Remove the share access if necessary
//
if ( SetShareAccess ) {
NwAcquireExclusiveFcb( Fcb->NonPagedFcb, TRUE ); IoRemoveShareAccess( FileObject, &Fcb->ShareAccess ); NwReleaseFcb( Fcb->NonPagedFcb ); }
//
// Failed to create
//
if ( Icb != NULL ) {
if ( Icb->Pid != 0 ) { NwUnmapPid(IrpContext->pNpScb, Icb->Pid, NULL ); } //
// dfergus 19 Apr 2001 #330484
//
NwDeleteIcb( NULL, Icb ); // added to fix 330484
IrpContext->Icb = NULL;
//
// if the operation failed, make sure we NULL out the
// FsContext field (nealch)
//
NwSetFileObject( FileObject, NULL, NULL ); }
//
// If this was a tree connect, derefence the extra
// reference on the VCB.
//
if ( CreateTreeConnection && !DeleteOnClose ) { if ( Vcb != NULL ) { NwDereferenceVcb( Vcb, IrpContext, FALSE ); } }
NwDequeueIrpContext( IrpContext, FALSE );
} else {
Icb->State = ICB_STATE_OPENED; if ( Fcb->State == FCB_STATE_OPEN_PENDING ) { Fcb->State = FCB_STATE_OPENED; }
if ( DeleteOnClose && !CreateTreeConnection ) { SetFlag( Fcb->Flags, FCB_FLAGS_DELETE_ON_CLOSE ); }
FileObject->SectionObjectPointer = &Fcb->NonPagedFcb->SegmentObject;
if ( MayBeADirectory ) {
//
// We successfully opened the file as a directory.
// If the DCB is newly created, it will be marked
// type FCB, update it.
//
Fcb->NodeTypeCode = NW_NTC_DCB; }
NwDequeueIrpContext( IrpContext, FALSE );
}
if ( OwnOpenLock ) { NwReleaseOpenLock( ); }
}
return( Iosb ); }
IO_STATUS_BLOCK ChangeDirectory( PIRP_CONTEXT IrpContext, PVCB Vcb, PICB Icb ) /*++
Routine Description:
This routines sets the directory for a remote drive.
Arguments:
IrpContext - Supplies all the information
Vcb - A pointer to the VCB for the remote drive.
Icb - A pointer to the file we are opening.
Return Value:
IO_STATUS_BLOCK - Status of operation
--*/ { IO_STATUS_BLOCK Iosb; PFCB Fcb; BYTE Attributes; BOOLEAN FirstTime = TRUE;
PAGED_CODE();
//
// No need to send a packet if we are opening the root of the volume.
//
if ( Icb->SuperType.Fcb->RelativeFileName.Length == 0 ) {
Iosb.Status = STATUS_SUCCESS; Iosb.Information = FILE_OPENED;
return( Iosb ); }
Retry:
if ( !BooleanFlagOn( Icb->SuperType.Fcb->Flags, FCB_FLAGS_LONG_NAME ) ) {
Iosb.Status = ExchangeWithWait ( IrpContext, SynchronousResponseCallback, "FwbbJ", NCP_SEARCH_FILE, -1, Vcb->Specific.Disk.Handle, SEARCH_ALL_DIRECTORIES, &Icb->SuperType.Fcb->RelativeFileName );
if ( NT_SUCCESS( Iosb.Status ) ) { Iosb.Status = ParseResponse( IrpContext, IrpContext->rsp, IrpContext->ResponseLength, "N==_b", 14, &Attributes ); }
} else {
Iosb.Status = ExchangeWithWait ( IrpContext, SynchronousResponseCallback, "LbbWDbDbC", NCP_LFN_GET_INFO, Vcb->Specific.Disk.LongNameSpace, Vcb->Specific.Disk.LongNameSpace, SEARCH_ALL_DIRECTORIES, LFN_FLAG_INFO_ATTRIBUTES | LFN_FLAG_INFO_MODIFY_TIME, Vcb->Specific.Disk.VolumeNumber, Vcb->Specific.Disk.Handle, 0, &Icb->SuperType.Fcb->RelativeFileName );
if ( NT_SUCCESS( Iosb.Status ) ) { Iosb.Status = ParseResponse( IrpContext, IrpContext->rsp, IrpContext->ResponseLength, "N_b", 4, &Attributes ); }
//
// Unfortunately, this succeeds even if the file in question
// is not a directory.
//
if ( NT_SUCCESS( Iosb.Status ) && ( !FlagOn( Attributes, NW_ATTRIBUTE_DIRECTORY ) ) ) {
Iosb.Status = STATUS_OBJECT_PATH_NOT_FOUND; } }
if ((Iosb.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, Vcb );
goto Retry; }
Fcb = Icb->SuperType.Fcb;
Fcb->NonPagedFcb->Attributes = (UCHAR)Attributes; SetFlag( Fcb->Flags, FCB_FLAGS_ATTRIBUTES_ARE_VALID );
//
// Set information field assuming success. It will be ignored
// if the NCP failed.
//
Iosb.Information = FILE_OPENED;
if ( Iosb.Status == STATUS_UNSUCCESSFUL ) { Iosb.Status = STATUS_OBJECT_PATH_NOT_FOUND; }
return( Iosb ); }
IO_STATUS_BLOCK CreateDir( PIRP_CONTEXT IrpContext, PVCB Vcb, PICB Icb ) /*++
Routine Description:
This routines create a new directory.
Arguments:
IrpContext - Supplies all the information
Vcb - A pointer to the VCB for the remote drive.
Return Value:
IO_STATUS_BLOCK - Status of operation
--*/ { IO_STATUS_BLOCK Iosb;
PAGED_CODE();
if ( Icb->SuperType.Fcb->RelativeFileName.Length == 0 ) { Iosb.Status = STATUS_ACCESS_DENIED; return( Iosb ); }
if ( !BooleanFlagOn( Icb->SuperType.Fcb->Flags, FCB_FLAGS_LONG_NAME ) ) {
if (!IsFatNameValid(&Icb->SuperType.Fcb->RelativeFileName)) {
Iosb.Status = STATUS_OBJECT_PATH_SYNTAX_BAD;
return( Iosb );
}
Iosb.Status = ExchangeWithWait ( IrpContext, SynchronousResponseCallback, "SbbJ", NCP_DIR_FUNCTION, NCP_CREATE_DIRECTORY, Vcb->Specific.Disk.Handle, 0xFF, &Icb->SuperType.Fcb->RelativeFileName );
} else {
Iosb.Status = ExchangeWithWait( IrpContext, SynchronousResponseCallback, "LbbWDDWbDbC", NCP_LFN_OPEN_CREATE, Vcb->Specific.Disk.LongNameSpace, LFN_FLAG_OM_CREATE, 0, // Search Flags,
0, // Return Info Mask
NW_ATTRIBUTE_DIRECTORY, 0x00ff, // Desired access
Vcb->Specific.Disk.VolumeNumber, Vcb->Specific.Disk.Handle, 0, // Short directory flag
&Icb->SuperType.Fcb->RelativeFileName );
}
if ( NT_SUCCESS( Iosb.Status ) ) { Iosb.Status = ParseResponse( IrpContext, IrpContext->rsp, IrpContext->ResponseLength, "N" ); }
//
// Set information field assuming success. It will be ignored
// if the NCP failed.
//
Iosb.Information = FILE_CREATED;
if ( Iosb.Status == STATUS_UNSUCCESSFUL ) { Iosb.Status = STATUS_OBJECT_NAME_COLLISION; }
return( Iosb ); }
NTSTATUS FileOrDirectoryExists( IN PIRP_CONTEXT IrpContext, IN PVCB Vcb, IN PICB Icb OPTIONAL, PUNICODE_STRING Name, OUT PBOOLEAN IsAFile ) /*++
Routine Description:
This routines looks to see if a file or directory exists.
Arguments:
IrpContext - Supplies allx the information
Vcb - A pointer to the VCB for the remote drive.
Icb - A pointer to the ICB for the file we are looking for.
Name - Fully qualified name.
IsAFile - Returns TRUE is the found file is a file, FALSE if it is a directory. Return nothing if the function returns FALSE.
Return Value:
IO_STATUS_BLOCK - Status of operation
--*/ { ULONG Attributes; ULONG FileSize; USHORT LastModifiedDate; USHORT LastModifiedTime; USHORT CreationDate; USHORT CreationTime = DEFAULT_TIME; USHORT LastAccessDate; NTSTATUS Status; PFCB Fcb; BOOLEAN FirstTime = TRUE;
PAGED_CODE();
//
// No need to send a packet if we are searching for the root of the volume.
//
if ( Name->Length == 0 ) { *IsAFile = FALSE;
return( STATUS_SUCCESS ); }
//
// Decide how to handle this request. If we have an ICB, use the FCB
// to determine the file name type, otherwise we have to make the
// decision here.
//
if ( Icb != NULL && !BooleanFlagOn( Icb->SuperType.Fcb->Flags, FCB_FLAGS_LONG_NAME ) ||
Vcb->Specific.Disk.LongNameSpace == LFN_NO_OS2_NAME_SPACE ||
IsFatNameValid( Name ) ) { Retry: //
// First try a file
//
IrpContext->ResponseLength = 0;
Status = ExchangeWithWait ( IrpContext, SynchronousResponseCallback, "FwbbJ", NCP_SEARCH_FILE, -1, Vcb->Specific.Disk.Handle, SEARCH_ALL_FILES, Name );
if ( NT_SUCCESS( Status ) ) { Status = ParseResponse( IrpContext, IrpContext->rsp, IrpContext->ResponseLength, "N==_b-dwwww", 14, &Attributes, &FileSize, &CreationDate, &LastAccessDate, &LastModifiedDate, &LastModifiedTime ); }
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, Vcb );
goto Retry; }
if ( Status == STATUS_UNSUCCESSFUL ) {
//
// Not a file, Is it a directory?
//
Status = ExchangeWithWait ( IrpContext, SynchronousResponseCallback, "FwbbJ", NCP_SEARCH_FILE, -1, Vcb->Specific.Disk.Handle, SEARCH_ALL_DIRECTORIES, Name );
if ( NT_SUCCESS( Status ) ) { Status = ParseResponse( IrpContext, IrpContext->rsp, IrpContext->ResponseLength, "N==_b", 14, &Attributes ); }
//
// If the exchange or ParseResponse fails then exit with not found
//
if ( !NT_SUCCESS( Status ) ) { return( STATUS_OBJECT_NAME_NOT_FOUND ); }
*IsAFile = FALSE; ASSERT( (Attributes & NW_ATTRIBUTE_DIRECTORY) != 0 );
} else {
if ( Status == STATUS_UNEXPECTED_NETWORK_ERROR && IrpContext->ResponseLength >= sizeof( NCP_RESPONSE ) ) {
//
// Work-around for netware bug. If netware returns short
// packet, just return success. We exit prematurely
// because we have no attributes to record.
//
Icb = NULL; *IsAFile = TRUE; return ( STATUS_SUCCESS ); }
if ( !NT_SUCCESS( Status ) ) { return( Status ); }
*IsAFile = TRUE; ASSERT( ( Attributes & NW_ATTRIBUTE_DIRECTORY ) == 0 );
}
} else {
Status = ExchangeWithWait ( IrpContext, SynchronousResponseCallback, "LbbWDbDbC", NCP_LFN_GET_INFO, Vcb->Specific.Disk.LongNameSpace, Vcb->Specific.Disk.LongNameSpace, SEARCH_ALL_DIRECTORIES, LFN_FLAG_INFO_ATTRIBUTES | LFN_FLAG_INFO_FILE_SIZE | LFN_FLAG_INFO_MODIFY_TIME | LFN_FLAG_INFO_CREATION_TIME, Vcb->Specific.Disk.VolumeNumber, Vcb->Specific.Disk.Handle, 0, Name );
if ( NT_SUCCESS( Status ) ) { Status = ParseResponse( IrpContext, IrpContext->rsp, IrpContext->ResponseLength, "N_e=e_xx_xx_x", 4, &Attributes, &FileSize, 6, &CreationTime, &CreationDate, 4, &LastModifiedTime, &LastModifiedDate, 4, &LastAccessDate ); }
//
// If the exchange or ParseResponse fails then exit with not found
//
if ( !NT_SUCCESS( Status ) ) { return( STATUS_OBJECT_NAME_NOT_FOUND ); }
if ( Attributes & NW_ATTRIBUTE_DIRECTORY) { *IsAFile = FALSE; } else { *IsAFile = TRUE; } }
//
// If the caller supplied an ICB, update the FCB attributes.
// We'll use this info if the caller does a query attributes
// on the ICB.
//
if ( Icb != NULL && *IsAFile ) {
Fcb = Icb->SuperType.Fcb; ASSERT( Fcb->NodeTypeCode == NW_NTC_FCB );
Fcb->NonPagedFcb->Attributes = (UCHAR)Attributes; Fcb->NonPagedFcb->Header.FileSize.QuadPart = FileSize; Fcb->LastModifiedDate = LastModifiedDate; Fcb->LastModifiedTime = LastModifiedTime; Fcb->CreationTime = CreationTime; Fcb->CreationDate = CreationDate; Fcb->LastAccessDate = LastAccessDate;
DebugTrace( 0, Dbg, "Attributes -> %08lx\n", Fcb->NonPagedFcb->Attributes ); DebugTrace( 0, Dbg, "FileSize.Low-> %08lx\n", Fcb->NonPagedFcb->Header.FileSize.LowPart ); DebugTrace( 0, Dbg, "ModifiedDate-> %08lx\n", Fcb->LastModifiedDate ); DebugTrace( 0, Dbg, "ModifiedTime-> %08lx\n", Fcb->LastModifiedTime ); DebugTrace( 0, Dbg, "CreationTime-> %08lx\n", Fcb->CreationTime ); DebugTrace( 0, Dbg, "CreationDate-> %08lx\n", Fcb->CreationDate ); DebugTrace( 0, Dbg, "LastAccDate -> %08lx\n", Fcb->LastAccessDate );
SetFlag( Fcb->Flags, FCB_FLAGS_ATTRIBUTES_ARE_VALID ); }
return( STATUS_SUCCESS ); }
IO_STATUS_BLOCK OpenFile( IN PIRP_CONTEXT IrpContext, IN PVCB Vcb, IN PICB Icb, IN BYTE Attributes, IN BYTE OpenFlags ) /*++
Routine Description:
This routines sets opens a file on a netware server. It fails if the file does not exist.
Arguments:
IrpContext - Supplies all the information
Vcb - A pointer to the VCB for the remote drive.
Icb - A pointer to the ICB we are opening.
Attributes - Open attributes.
OpenFlags - Open mode and sharing mode flags.
Return Value:
IO_STATUS_BLOCK - Status of operation
--*/ { IO_STATUS_BLOCK Iosb; PFCB Fcb;
PAGED_CODE();
//
// No need to send a packet if we are trying to open the root of
// the volume as a file.
//
if ( Icb->SuperType.Fcb->RelativeFileName.Length == 0 ) { Iosb.Status = STATUS_FILE_IS_A_DIRECTORY; return( Iosb ); }
Fcb = Icb->SuperType.Fcb; ASSERT( NodeType( Fcb ) == NW_NTC_FCB );
//
// Send the open request and wait for the response.
//
if ( !BooleanFlagOn( Fcb->Flags, FCB_FLAGS_LONG_NAME ) ) {
Iosb.Status = ExchangeWithWait ( IrpContext, SynchronousResponseCallback, "FbbbJ", NCP_OPEN_FILE, Vcb->Specific.Disk.Handle, SEARCH_ALL_FILES, OpenFlags, &Icb->SuperType.Fcb->RelativeFileName );
if ( ( ReadExecOnlyFiles ) && ( !NT_SUCCESS( Iosb.Status ) ) ) {
//
// Retry the open with the appropriate flags for
// execute only files.
//
Iosb.Status = ExchangeWithWait ( IrpContext, SynchronousResponseCallback, "FbbbJ", NCP_OPEN_FILE, Vcb->Specific.Disk.Handle, SEARCH_EXEC_ONLY_FILES, OpenFlags, &Icb->SuperType.Fcb->RelativeFileName ); }
if ( NT_SUCCESS( Iosb.Status ) ) { Iosb.Status = ParseResponse( IrpContext, IrpContext->rsp, IrpContext->ResponseLength, "Nr=_b-dwwww", Icb->Handle, sizeof( Icb->Handle ), 14, &Fcb->NonPagedFcb->Attributes, &Fcb->NonPagedFcb->Header.FileSize, &Fcb->CreationDate, &Fcb->LastAccessDate, &Fcb->LastModifiedDate, &Fcb->LastModifiedTime );
Fcb->CreationTime = DEFAULT_TIME;
}
} else {
Iosb.Status = ExchangeWithWait ( IrpContext, SynchronousResponseCallback, "LbbWDDWbDbC", NCP_LFN_OPEN_CREATE, Vcb->Specific.Disk.LongNameSpace, LFN_FLAG_OM_OPEN, NW_ATTRIBUTE_HIDDEN | NW_ATTRIBUTE_SYSTEM, // Search Flags,
LFN_FLAG_INFO_ATTRIBUTES | LFN_FLAG_INFO_FILE_SIZE | LFN_FLAG_INFO_MODIFY_TIME | LFN_FLAG_INFO_CREATION_TIME, 0, // Create attributes
OpenFlags, // Desired access
Vcb->Specific.Disk.VolumeNumber, Vcb->Specific.Disk.Handle, 0, // Short directory flag
&Icb->SuperType.Fcb->RelativeFileName );
if ( ( ReadExecOnlyFiles ) && ( !NT_SUCCESS( Iosb.Status ) ) ) { Iosb.Status = ExchangeWithWait ( IrpContext, SynchronousResponseCallback, "LbbWDDWbDbC", NCP_LFN_OPEN_CREATE, Vcb->Specific.Disk.LongNameSpace, LFN_FLAG_OM_OPEN, NW_ATTRIBUTE_EXEC_ONLY, LFN_FLAG_INFO_ATTRIBUTES | LFN_FLAG_INFO_FILE_SIZE | LFN_FLAG_INFO_MODIFY_TIME | LFN_FLAG_INFO_CREATION_TIME, 0, // Create attributes
OpenFlags, // Desired access
Vcb->Specific.Disk.VolumeNumber, Vcb->Specific.Disk.Handle, 0, // Short directory flag
&Icb->SuperType.Fcb->RelativeFileName ); }
if ( NT_SUCCESS( Iosb.Status ) ) { Iosb.Status = ParseResponse( IrpContext, IrpContext->rsp, IrpContext->ResponseLength, "Ne_e=e_xx_xx_x", &Icb->Handle[2], 6, &Fcb->NonPagedFcb->Attributes, &Fcb->NonPagedFcb->Header.FileSize, 6, &Fcb->CreationTime, &Fcb->CreationDate, 4, &Fcb->LastModifiedTime, &Fcb->LastModifiedDate, 4, &Fcb->LastAccessDate ); } }
if ( NT_SUCCESS( Iosb.Status ) ) {
//
// NT does not allow you to open a read only file for write access.
// Netware does. To fake NT semantics, check to see if we should
// fail the open that the netware server just succeeded.
//
if ( ( Fcb->NonPagedFcb->Attributes & NW_ATTRIBUTE_READ_ONLY ) && ( OpenFlags & NW_OPEN_FOR_WRITE ) ) {
CloseFile( IrpContext, Icb ); Iosb.Status = STATUS_ACCESS_DENIED; }
SetFlag( Fcb->Flags, FCB_FLAGS_ATTRIBUTES_ARE_VALID ); Icb->HasRemoteHandle = TRUE;
DebugTrace( 0, Dbg, "Attributes -> %08lx\n", Fcb->NonPagedFcb->Attributes ); DebugTrace( 0, Dbg, "FileSize.Low-> %08lx\n", Fcb->NonPagedFcb->Header.FileSize.LowPart ); DebugTrace( 0, Dbg, "ModifiedDate-> %08lx\n", Fcb->LastModifiedDate ); DebugTrace( 0, Dbg, "ModifiedTime-> %08lx\n", Fcb->LastModifiedTime ); DebugTrace( 0, Dbg, "CreationDate-> %08lx\n", Fcb->CreationDate ); DebugTrace( 0, Dbg, "CreationTime-> %08lx\n", Fcb->CreationTime ); DebugTrace( 0, Dbg, "LastAccDate -> %08lx\n", Fcb->LastAccessDate );
}
//
// Set information field assuming success. It will be ignored
// if the NCP failed.
//
Iosb.Information = FILE_OPENED;
if ( Iosb.Status == STATUS_UNSUCCESSFUL ) { Iosb.Status = STATUS_OBJECT_NAME_NOT_FOUND; }
return( Iosb ); }
IO_STATUS_BLOCK CreateNewFile( IN PIRP_CONTEXT IrpContext, IN PVCB Vcb, IN PICB Icb, IN BYTE CreateAttributes, IN BYTE OpenFlags ) /*++
Routine Description:
This routines creates a new file on a netware server. It fails if the file exists.
Arguments:
IrpContext - Supplies all the information
Vcb - A pointer to the VCB for the remote drive.
Icb - A pointer to the ICB we are opening.
CreateAttributes - Create attributes.
OpenFlags - Open mode and sharing mode flags.
Return Value:
IO_STATUS_BLOCK - Status of operation
--*/ { IO_STATUS_BLOCK Iosb; PFCB Fcb; UCHAR DelayedAttributes; BOOLEAN CloseAndReopen;
PAGED_CODE();
//
// If the user opens the file for shared access, then we will need to
// create the file close, then reopen it (since we have no NCP to say
// create with shared access). If the file is being created read-only,
// and the creator requests write access then we pull the additional
// trick of creating the file without the read-only, and set it later,
// so that the second open can succeed.
//
CloseAndReopen = FALSE; DelayedAttributes = 0;
if ( OpenFlags != NW_OPEN_EXCLUSIVE ) { CloseAndReopen = TRUE;
if ( ( CreateAttributes & NW_ATTRIBUTE_READ_ONLY ) && ( OpenFlags & NW_OPEN_FOR_WRITE ) ) {
DelayedAttributes = CreateAttributes; CreateAttributes = 0; } }
//
// Send the create request and wait for the response.
//
Fcb = Icb->SuperType.Fcb;
if ( !BooleanFlagOn( Fcb->Flags, FCB_FLAGS_LONG_NAME ) ) {
if (!IsFatNameValid(&Icb->SuperType.Fcb->RelativeFileName)) {
Iosb.Status = STATUS_OBJECT_PATH_SYNTAX_BAD;
return( Iosb );
}
Iosb.Status = ExchangeWithWait ( IrpContext, SynchronousResponseCallback, "FbbJ", // NCP Create New File
NCP_CREATE_NEW_FILE, Vcb->Specific.Disk.Handle, CreateAttributes, &Icb->SuperType.Fcb->RelativeFileName );
if ( NT_SUCCESS( Iosb.Status ) ) { Iosb.Status = ParseResponse( IrpContext, IrpContext->rsp, IrpContext->ResponseLength, "Nr=_b-dwwww", Icb->Handle, sizeof( Icb->Handle ), 14, &Fcb->NonPagedFcb->Attributes, &Fcb->NonPagedFcb->Header.FileSize, &Fcb->CreationDate, &Fcb->LastAccessDate, &Fcb->LastModifiedDate, &Fcb->LastModifiedTime );
Fcb->CreationTime = DEFAULT_TIME;
}
} else {
Iosb.Status = ExchangeWithWait ( IrpContext, SynchronousResponseCallback, "LbbWDDWbDbC", NCP_LFN_OPEN_CREATE, Vcb->Specific.Disk.LongNameSpace, LFN_FLAG_OM_CREATE, 0, // Search Flags
LFN_FLAG_INFO_ATTRIBUTES | LFN_FLAG_INFO_FILE_SIZE | LFN_FLAG_INFO_MODIFY_TIME | LFN_FLAG_INFO_CREATION_TIME, CreateAttributes, 0, // Desired access
Vcb->Specific.Disk.VolumeNumber, Vcb->Specific.Disk.Handle, 0, // Short directory flag
&Icb->SuperType.Fcb->RelativeFileName );
if ( NT_SUCCESS( Iosb.Status ) ) { Iosb.Status = ParseResponse( IrpContext, IrpContext->rsp, IrpContext->ResponseLength, "Ne_e=e_xx_xx_x", &Icb->Handle[2], 6, &Fcb->NonPagedFcb->Attributes, &Fcb->NonPagedFcb->Header.FileSize, 6, &Fcb->CreationTime, &Fcb->CreationDate, 4, &Fcb->LastModifiedTime, &Fcb->LastModifiedDate, 4, &Fcb->LastAccessDate ); } }
if ( NT_SUCCESS( Iosb.Status ) ) { SetFlag( Fcb->Flags, FCB_FLAGS_ATTRIBUTES_ARE_VALID ); Icb->HasRemoteHandle = TRUE; DebugTrace( 0, Dbg, "Attributes -> %08lx\n", Fcb->NonPagedFcb->Attributes ); DebugTrace( 0, Dbg, "FileSize.Low-> %08lx\n", Fcb->NonPagedFcb->Header.FileSize.LowPart ); DebugTrace( 0, Dbg, "ModifiedDate-> %08lx\n", Fcb->LastModifiedDate ); DebugTrace( 0, Dbg, "ModifiedTime-> %08lx\n", Fcb->LastModifiedTime ); DebugTrace( 0, Dbg, "CreationDate-> %08lx\n", Fcb->CreationDate ); DebugTrace( 0, Dbg, "CreationTime-> %08lx\n", Fcb->CreationTime ); DebugTrace( 0, Dbg, "LastAcceDate-> %08lx\n", Fcb->LastAccessDate ); }
if ( Iosb.Status == STATUS_UNSUCCESSFUL ) { Iosb.Status = STATUS_OBJECT_NAME_COLLISION; }
if ( !NT_SUCCESS( Iosb.Status ) ) { return( Iosb ); }
//
// We've created the file, and the users wants shared access to the
// file. Close the file and reopen in sharing mode.
//
if ( CloseAndReopen ) { CloseFile( IrpContext, Icb ); Iosb = OpenFile( IrpContext, Vcb, Icb, CreateAttributes, OpenFlags ); }
//
// If we need to set attributes, do it now. Ignore errors, if any.
//
if ( DelayedAttributes != 0 ) {
ExchangeWithWait( IrpContext, SynchronousResponseCallback, "FbbbU", NCP_SET_FILE_ATTRIBUTES, DelayedAttributes, Fcb->Vcb->Specific.Disk.Handle, SEARCH_ALL_FILES, &Fcb->RelativeFileName );
}
//
// Set information field assuming success. It will be ignored
// if the NCP failed.
//
Iosb.Information = FILE_CREATED; return( Iosb ); }
IO_STATUS_BLOCK CreateOrOverwriteFile( IN PIRP_CONTEXT IrpContext, IN PVCB Vcb, IN PICB Icb, IN BYTE CreateAttributes, IN BYTE OpenFlags, IN BOOLEAN CreateOperation ) /*++
Routine Description:
This routines creates a file on a netware server. If the file exists it is overwritten.
Arguments:
IrpContext - Supplies all the information
Vcb - A pointer to the VCB for the remote drive.
Icb - A pointer to the ICB we are opening.
Attributes - Open attributes.
OpenFlags - Open mode and sharing mode flags.
Return Value:
IO_STATUS_BLOCK - Status of operation
--*/ { IO_STATUS_BLOCK Iosb; PFCB Fcb; UCHAR DelayedAttributes; BOOLEAN CloseAndReopen;
PAGED_CODE();
Fcb = Icb->SuperType.Fcb;
//
// Send the request and wait for the response.
//
if ( !BooleanFlagOn( Fcb->Flags, FCB_FLAGS_LONG_NAME ) ) {
if (!IsFatNameValid(&Icb->SuperType.Fcb->RelativeFileName)) {
Iosb.Status = STATUS_OBJECT_PATH_SYNTAX_BAD;
return( Iosb );
}
//
// If the user opens the file for shared access, then we will need to
// create the file close, then reopen it (since we have no NCP to say
// create with shared access). If the file is being created read-only,
// and the creator requests write access then we pull the additional
// trick of creating the file without the read-only, and set it later,
// so that the second open can succeed.
//
if ( ( CreateAttributes & NW_ATTRIBUTE_READ_ONLY ) && ( OpenFlags & NW_OPEN_FOR_WRITE ) ) {
DelayedAttributes = CreateAttributes; CreateAttributes = 0; } else { DelayedAttributes = 0; }
//
// Dos namespace create always returns the file exclusive.
//
if (!FlagOn(OpenFlags, NW_OPEN_EXCLUSIVE)) { CloseAndReopen = TRUE; } else { CloseAndReopen = FALSE; }
Iosb.Status = ExchangeWithWait ( IrpContext, SynchronousResponseCallback, "FbbJ", NCP_CREATE_FILE, Vcb->Specific.Disk.Handle, CreateAttributes, &Icb->SuperType.Fcb->RelativeFileName );
if ( NT_SUCCESS( Iosb.Status ) ) { Iosb.Status = ParseResponse( IrpContext, IrpContext->rsp, IrpContext->ResponseLength, "Nr=_b-dwwww", Icb->Handle, sizeof( Icb->Handle ), 14, &Fcb->NonPagedFcb->Attributes, &Fcb->NonPagedFcb->Header.FileSize, &Fcb->CreationDate, &Fcb->LastAccessDate, &Fcb->LastModifiedDate, &Fcb->LastModifiedTime );
Fcb->CreationTime = DEFAULT_TIME;
}
//
// We've created the file, and the users wants shared access to the
// file. Close the file and reopen in sharing mode.
//
if (( NT_SUCCESS( Iosb.Status ) ) && ( CloseAndReopen )) {
CloseFile( IrpContext, Icb ); Iosb = OpenFile( IrpContext, Vcb, Icb, CreateAttributes, OpenFlags ); }
if ( DelayedAttributes != 0 ) { ExchangeWithWait( IrpContext, SynchronousResponseCallback, "FbbbU", NCP_SET_FILE_ATTRIBUTES, DelayedAttributes, Fcb->Vcb->Specific.Disk.Handle, SEARCH_ALL_FILES, &Fcb->RelativeFileName ); }
} else {
Iosb.Status = ExchangeWithWait ( IrpContext, SynchronousResponseCallback, "LbbWDDWbDbC", NCP_LFN_OPEN_CREATE, Vcb->Specific.Disk.LongNameSpace, LFN_FLAG_OM_OVERWRITE, 0, // Search Flags
LFN_FLAG_INFO_ATTRIBUTES | LFN_FLAG_INFO_FILE_SIZE | LFN_FLAG_INFO_MODIFY_TIME | LFN_FLAG_INFO_CREATION_TIME, CreateAttributes, OpenFlags, // DesiredAccess
Vcb->Specific.Disk.VolumeNumber, Vcb->Specific.Disk.Handle, 0, // Short directory flag
&Icb->SuperType.Fcb->RelativeFileName );
if ( NT_SUCCESS( Iosb.Status ) ) { Iosb.Status = ParseResponse( IrpContext, IrpContext->rsp, IrpContext->ResponseLength, "Ne_e=e_xx_xx_x", &Icb->Handle[2], 6, &Fcb->NonPagedFcb->Attributes, &Fcb->NonPagedFcb->Header.FileSize, 6, &Fcb->CreationTime, &Fcb->CreationDate, 4, &Fcb->LastModifiedTime, &Fcb->LastModifiedDate, 4, &Fcb->LastAccessDate ); } }
if ( NT_SUCCESS( Iosb.Status ) ) { SetFlag( Fcb->Flags, FCB_FLAGS_ATTRIBUTES_ARE_VALID ); Icb->HasRemoteHandle = TRUE; DebugTrace( 0, Dbg, "Attributes -> %08lx\n", Fcb->NonPagedFcb->Attributes ); DebugTrace( 0, Dbg, "FileSize.Low-> %08lx\n", Fcb->NonPagedFcb->Header.FileSize.LowPart ); DebugTrace( 0, Dbg, "ModifiedDate-> %08lx\n", Fcb->LastModifiedDate ); DebugTrace( 0, Dbg, "ModifiedTime-> %08lx\n", Fcb->LastModifiedTime ); DebugTrace( 0, Dbg, "CreationDate-> %08lx\n", Fcb->CreationDate ); DebugTrace( 0, Dbg, "CreationTime-> %08lx\n", Fcb->CreationTime ); DebugTrace( 0, Dbg, "LastAccDate -> %08lx\n", Fcb->LastAccessDate ); } else { return( Iosb ); }
//
// Set information field assuming success. It will be ignored
// if the NCP failed.
//
if ( CreateOperation) { Iosb.Information = FILE_CREATED; } else { Iosb.Information = FILE_OVERWRITTEN; }
return( Iosb ); }
IO_STATUS_BLOCK OpenRenameTarget( IN PIRP_CONTEXT IrpContext, IN PVCB Vcb, IN PDCB Dcb, IN PICB* Icb ) /*++
Routine Description:
This routine opens a directory. If the filename provided specifies a directory then the file/directory to be renamed will be put in this directory.
If the target foo\bar does not exist or is a file then the source of the rename must be a file and will end up in the directory foo with the name bar
Arguments:
IrpContext - Supplies all the information
Vcb - A pointer to the VCB for the remote drive.
Dcb - A pointer to the DCB for relative opens. If NULL the FileName is an full path name. If non NUL the FileName is relative to this directory.
Icb - A pointer to where the address of the Icb is to be stored.
Return Value:
NT_STATUS - Status of operation
--*/ { PIRP Irp; PIO_STACK_LOCATION IrpSp;
IO_STATUS_BLOCK Iosb; PFCB Fcb; BOOLEAN FullNameIsAFile; BOOLEAN FullNameExists; BOOLEAN PathIsAFile;
#if 0
UNICODE_STRING Drive; WCHAR DriveLetter; UNICODE_STRING Server; UNICODE_STRING Volume; UNICODE_STRING FileName; #endif
UNICODE_STRING Path; UNICODE_STRING FullName; UNICODE_STRING CompleteName; UNICODE_STRING VcbName; PWCH pTrailingSlash;
USHORT i; USHORT DcbNameLength;
PAGED_CODE();
DebugTrace(+1, Dbg, "OpenRenameTarget\n", 0);
//
// Get the current IRP stack location
//
Irp = IrpContext->pOriginalIrp;
IrpSp = IoGetCurrentIrpStackLocation( Irp );
//
// Build a complete filename of the form \g:\server\volume\dir1\file
//
if ( Dcb != NULL ) {
//
// Strip to UID portion of the DCB name.
//
for ( i = 0 ; i < Dcb->FullFileName.Length / sizeof(WCHAR) ; i++ ) { if ( Dcb->FullFileName.Buffer[i] == OBJ_NAME_PATH_SEPARATOR ) { break; } }
ASSERT( Dcb->FullFileName.Buffer[i] == OBJ_NAME_PATH_SEPARATOR );
//
// Now build the full name by appending the file name to the DCB name.
//
DcbNameLength = Dcb->FullFileName.Length - ( i * sizeof(WCHAR) ); CompleteName.Length = DcbNameLength + IrpSp->FileObject->FileName.Length + sizeof( WCHAR); CompleteName.MaximumLength = CompleteName.Length; CompleteName.Buffer = ALLOCATE_POOL_EX( PagedPool, CompleteName.Length );
RtlCopyMemory( CompleteName.Buffer, Dcb->FullFileName.Buffer + i, DcbNameLength );
CompleteName.Buffer[ DcbNameLength / sizeof(WCHAR) ] = L'\\';
RtlCopyMemory( CompleteName.Buffer + DcbNameLength / sizeof(WCHAR ) + 1, IrpSp->FileObject->FileName.Buffer, IrpSp->FileObject->FileName.Length );
Dcb = NULL;
} else {
CompleteName = IrpSp->FileObject->FileName;
}
//
// Calculate the VCB name, without the UID prefix.
//
VcbName.Buffer = wcschr( Vcb->Name.Buffer, L'\\' ); VcbName.Length = (USHORT) (Vcb->Name.Length - ( (PCHAR)VcbName.Buffer - (PCHAR)Vcb->Name.Buffer ));
//
// Calculate the target relative name. This is simply the complete
// name minus the VcbName and the leading backslash.
//
FullName.Buffer = CompleteName.Buffer + ( VcbName.Length / sizeof(WCHAR) ) + 1; FullName.Length = (USHORT) (CompleteName.Length - ( (PCHAR)FullName.Buffer - (PCHAR)CompleteName.Buffer ));
//
// Calculate the target directory relative name. This the the target
// full name, minus the last component of the name.
//
pTrailingSlash = FullName.Buffer + FullName.Length / sizeof(WCHAR) - 1; for ( i = 0; i < FullName.Length ; i += sizeof(WCHAR) ) { if ( *pTrailingSlash == L'\\' ) { break; } --pTrailingSlash; }
Path.Buffer = FullName.Buffer;
if ( i == FullName.Length ) {
//
// If no trailing slash was found, the the target path is the
// root directory.
//
Path.Length = 0;
} else {
Path.Length = (USHORT) ((PCHAR)pTrailingSlash - (PCHAR)FullName.Buffer);
}
#if 0
Iosb.Status = CrackPath( &CompleteName, &Drive, &DriveLetter, &Server, &Volume, &Path, &FileName, &FullName ); #endif
Iosb.Status = FileOrDirectoryExists( IrpContext, Vcb, NULL, &Path, &PathIsAFile );
if ( !NT_SUCCESS( Iosb.Status) ) {
// The directory containing the file does not exist
return(Iosb); }
Iosb.Status = FileOrDirectoryExists( IrpContext, Vcb, NULL, &FullName, &FullNameIsAFile );
if ( !NT_SUCCESS( Iosb.Status ) ) { FullNameExists = FALSE; Iosb.Information = FILE_DOES_NOT_EXIST; } else { FullNameExists = TRUE; Iosb.Information = 0; }
DebugTrace( 0, Dbg, "FullNameExists = %08lx\n", FullNameExists); DebugTrace( 0, Dbg, "FullNameIsAFile = %08lx\n", FullNameIsAFile);
try { UNICODE_STRING TargetPath;
//
// Find the FCB for this file. If the FCB exists, we get a
// referenced pointer. Otherwise a new FCB is created.
// The file is the complete path minus the target filename.
//
TargetPath = CompleteName;
Fcb = NwFindFcb( IrpContext->pScb, Vcb, &TargetPath, Dcb ); // in rare cases, NwFindFcb might return NULL instead of throwing an exception
// Raid # 432500
if (Fcb == NULL) { DebugTrace(0, Dbg, "NwFindFcb returned NULL in OpenRenameTarget\n", 0); Iosb.Status = STATUS_INVALID_PARAMETER; try_return( Iosb ); }
//
// Now create the ICB.
//
*Icb = NwCreateIcb( NW_NTC_ICB, Fcb );
(*Icb)->FileObject = IrpSp->FileObject; NwSetFileObject( IrpSp->FileObject, Fcb->NonPagedFcb, *Icb ); (*Icb)->Exists = FullNameExists; (*Icb)->IsAFile = FullNameIsAFile;
try_return(Iosb.Status = STATUS_SUCCESS);
try_exit: NOTHING;
} finally {
if ( AbnormalTermination() || !NT_SUCCESS( Iosb.Status ) ) {
//
// Failed to create
//
if ( *Icb != NULL ) { NwDeleteIcb( NULL, *Icb ); *Icb = NULL; } } }
DebugTrace(-1, Dbg, "OpenRenameTarget\n", Iosb.Status);
return( Iosb ); }
IO_STATUS_BLOCK CreatePrintJob( PIRP_CONTEXT IrpContext, PVCB Vcb, PICB Icb, PUNICODE_STRING DriveName ) /*++
Routine Description:
This routines create a new directory.
Arguments:
IrpContext - Supplies all the information
Vcb - A pointer to the VCB for the remote print queue.
Icb - A pointer to the newly created ICB.
DriveName - LPTx
Return Value:
IO_STATUS_BLOCK - Status of operation
--*/ { IO_STATUS_BLOCK Iosb; PFCB Fcb; ANSI_STRING ODriveName; static CHAR LptName[] = "LPT" ; ULONG PrintOptions; PLOGON Logon; PUNICODE_STRING BannerName;
PAGED_CODE();
BannerName = &IrpContext->pScb->UserName; NwAcquireExclusiveRcb( &NwRcb, TRUE ); Logon = FindUser( &IrpContext->pScb->UserUid, TRUE ); if ( Logon == NULL ) { PrintOptions = NwPrintOptions; } else { PrintOptions = Logon->NwPrintOptions; /*
* If user name is GUEST, use the validated user name */ if ((BannerName->Length == 0 ) || (RtlCompareUnicodeString( BannerName, &Guest.UserName, TRUE ) == 0 )) { BannerName = &Logon->UserName; } } NwReleaseRcb( &NwRcb );
//
// Make sure the print queue name is correct.
//
if ( Icb->SuperType.Fcb->RelativeFileName.Length != 0 ) { Iosb.Status = STATUS_OBJECT_PATH_SYNTAX_BAD; return( Iosb ); }
//
// Send a create queue job packet, and wait the response.
//
if ((DriveName->Length == 0 ) || (!NT_SUCCESS(RtlUnicodeStringToOemString( &ODriveName, DriveName, TRUE )))) { //
// if we dont have a name, use the string "LPT". we do this because
// some printers insist on a name.
//
RtlInitString(&ODriveName, LptName); }
Iosb.Status = ExchangeWithWait ( IrpContext, SynchronousResponseCallback, "Sd_ddw_b_r_bbwwww_x-x_", // Format string
NCP_ADMIN_FUNCTION, NCP_CREATE_QUEUE_JOB, Vcb->Specific.Print.QueueId,// Queue ID
6, // Skip bytes
0xffffffff, // Target Server ID number
0xffffffff, 0xffff, // Target Execution time
11, // Skip bytes
0x00, // Job Control Flags
26, // Skip bytes
ODriveName.Buffer, ODriveName.Length, // Description
50 - ODriveName.Length , // Description pad
0, // Version number
8, // Tab Size
1, // Number of copies
PrintOptions, // Control Flags
0x3C, // Maximum lines
0x84, // Maximum characters
22, // Skip bytes
BannerName, 13, // Banner Name
&Vcb->ShareName, 12, // Header Name
1+14+80 // null last string & skip rest of client area
);
//
// free the string if it was allocated
//
if (ODriveName.Buffer != LptName) RtlFreeAnsiString(&ODriveName);
if ( NT_SUCCESS( Iosb.Status ) ) { Iosb.Status = ParseResponse( IrpContext, IrpContext->rsp, IrpContext->ResponseLength, "N_w_r", 22, &Icb->JobId, 18, Icb->Handle, sizeof(Icb->Handle) );
}
if ( NT_SUCCESS( Iosb.Status ) ) {
Fcb = Icb->SuperType.Fcb;
Fcb->NonPagedFcb->Attributes = 0; Fcb->CreationDate = 0; Fcb->LastAccessDate = 0; Fcb->LastModifiedDate = 0; Fcb->LastModifiedTime = 0;
Icb->HasRemoteHandle = TRUE; Icb->IsPrintJob = TRUE; Icb->ActuallyPrinted = FALSE;
SetFlag( Fcb->Flags, FCB_FLAGS_ATTRIBUTES_ARE_VALID );
}
//
// Set information field assuming success. It will be ignored
// if the NCP failed.
//
Iosb.Information = FILE_CREATED;
if ( Iosb.Status == STATUS_UNSUCCESSFUL ) { Iosb.Status = STATUS_OBJECT_NAME_COLLISION; }
return( Iosb ); }
VOID CloseFile( PIRP_CONTEXT pIrpContext, PICB pIcb ) /*++
Routine Description:
This routines closes an opened file.
Arguments:
pIrpContext - Supplies all the information
pIcb - A pointer to the newly created ICB.
Return Value:
None.
--*/ { PAGED_CODE();
ExchangeWithWait( pIrpContext, SynchronousResponseCallback, "F-r", NCP_CLOSE, pIcb->Handle, 6 ); }
|