mirror of
https://github.com/winfsp/winfsp.git
synced 2025-04-25 09:52:23 -05:00
367 lines
13 KiB
C
367 lines
13 KiB
C
/**
|
|
* @file dll/create.c
|
|
*
|
|
* @copyright 2015 Bill Zissimopoulos
|
|
*/
|
|
|
|
#include <dll/library.h>
|
|
|
|
static inline
|
|
BOOLEAN FspIsRootDirectory(PWSTR FileName)
|
|
{
|
|
for (PWSTR Pointer = FileName; *Pointer; Pointer++)
|
|
if (L'\\' != *Pointer)
|
|
return FALSE;
|
|
return TRUE;
|
|
}
|
|
|
|
static inline
|
|
NTSTATUS FspCreateCheck(FSP_FILE_SYSTEM *FileSystem,
|
|
FSP_FSCTL_TRANSACT_REQ *Request, BOOLEAN AllowTraverseCheck,
|
|
PDWORD PGrantedAccess)
|
|
{
|
|
NTSTATUS Result;
|
|
PWSTR Path, Suffix;
|
|
|
|
FspPathSuffix((PWSTR)Request->Buffer, &Path, &Suffix);
|
|
Result = FspAccessCheck(FileSystem, Request, TRUE,
|
|
(Request->Req.Create.CreateOptions & FILE_DIRECTORY_FILE) ?
|
|
FILE_ADD_SUBDIRECTORY : FILE_ADD_FILE,
|
|
PGrantedAccess);
|
|
FspPathCombine((PWSTR)Request->Buffer, Suffix);
|
|
|
|
if (NT_SUCCESS(Result))
|
|
*PGrantedAccess = (MAXIMUM_ALLOWED & Request->Req.Create.DesiredAccess) ?
|
|
FILE_ALL_ACCESS : Request->Req.Create.DesiredAccess;
|
|
|
|
return Result;
|
|
}
|
|
|
|
static NTSTATUS FspFileSystemOpCreate_FileCreate(FSP_FILE_SYSTEM *FileSystem,
|
|
FSP_FSCTL_TRANSACT_REQ *Request)
|
|
{
|
|
if (FspIsRootDirectory((PWSTR)Request->Buffer))
|
|
return STATUS_ACCESS_DENIED;
|
|
|
|
NTSTATUS Result;
|
|
DWORD GrantedAccess;
|
|
FSP_FILE_NODE *FileNode;
|
|
FSP_FSCTL_TRANSACT_RSP Response;
|
|
|
|
Result = FspCreateCheck(FileSystem, Request, TRUE, &GrantedAccess);
|
|
if (!NT_SUCCESS(Result))
|
|
return FspFileSystemSendResponseWithStatus(FileSystem, Request, Result);
|
|
|
|
Result = FileSystem->Interface->FileCreate(FileSystem, Request, &FileNode);
|
|
if (!NT_SUCCESS(Result))
|
|
return FspFileSystemSendResponseWithStatus(FileSystem, Request, Result);
|
|
|
|
FileNode->Flags = 0;
|
|
memset(&FileNode->ShareAccess, 0, sizeof FileNode->ShareAccess);
|
|
|
|
FspShareCheck(FileSystem, GrantedAccess, Request->Req.Create.ShareAccess, FileNode);
|
|
|
|
memset(&Response, 0, sizeof Response);
|
|
Response.Size = sizeof Response;
|
|
Response.Kind = Request->Kind;
|
|
Response.Hint = Request->Hint;
|
|
Response.IoStatus.Status = STATUS_SUCCESS;
|
|
Response.IoStatus.Information = FILE_CREATED;
|
|
Response.Rsp.Create.Opened.UserContext = (UINT_PTR)FileNode;
|
|
Response.Rsp.Create.Opened.GrantedAccess = GrantedAccess;
|
|
return FspFileSystemSendResponse(FileSystem, &Response);
|
|
}
|
|
|
|
static NTSTATUS FspFileSystemOpCreate_FileOpen(FSP_FILE_SYSTEM *FileSystem,
|
|
FSP_FSCTL_TRANSACT_REQ *Request)
|
|
{
|
|
NTSTATUS Result;
|
|
DWORD GrantedAccess;
|
|
FSP_FILE_NODE *FileNode;
|
|
FSP_FSCTL_TRANSACT_RSP Response;
|
|
|
|
Result = FspAccessCheck(FileSystem, Request, TRUE,
|
|
Request->Req.Create.DesiredAccess, &GrantedAccess);
|
|
if (!NT_SUCCESS(Result))
|
|
return FspFileSystemSendResponseWithStatus(FileSystem, Request, Result);
|
|
|
|
Result = FileSystem->Interface->FileOpen(FileSystem, Request, &FileNode);
|
|
if (!NT_SUCCESS(Result))
|
|
return FspFileSystemSendResponseWithStatus(FileSystem, Request, Result);
|
|
|
|
Result = FspShareCheck(FileSystem, GrantedAccess, Request->Req.Create.ShareAccess, FileNode);
|
|
if (!NT_SUCCESS(Result))
|
|
{
|
|
if (0 != FileSystem->Interface->FileClose)
|
|
FileSystem->Interface->FileClose(FileSystem, Request, FileNode);
|
|
return FspFileSystemSendResponseWithStatus(FileSystem, Request, Result);
|
|
}
|
|
|
|
memset(&Response, 0, sizeof Response);
|
|
Response.Size = sizeof Response;
|
|
Response.Kind = Request->Kind;
|
|
Response.Hint = Request->Hint;
|
|
Response.IoStatus.Status = STATUS_SUCCESS;
|
|
Response.IoStatus.Information = FILE_OPENED;
|
|
Response.Rsp.Create.Opened.UserContext = (UINT_PTR)FileNode;
|
|
Response.Rsp.Create.Opened.GrantedAccess = GrantedAccess;
|
|
return FspFileSystemSendResponse(FileSystem, &Response);
|
|
}
|
|
|
|
static NTSTATUS FspFileSystemOpCreate_FileOpenIf(FSP_FILE_SYSTEM *FileSystem,
|
|
FSP_FSCTL_TRANSACT_REQ *Request)
|
|
{
|
|
NTSTATUS Result;
|
|
DWORD GrantedAccess;
|
|
FSP_FILE_NODE *FileNode;
|
|
FSP_FSCTL_TRANSACT_RSP Response;
|
|
BOOLEAN Create = FALSE;
|
|
|
|
Result = FspAccessCheck(FileSystem, Request, TRUE,
|
|
Request->Req.Create.DesiredAccess, &GrantedAccess);
|
|
if (!NT_SUCCESS(Result))
|
|
{
|
|
if (STATUS_OBJECT_NAME_NOT_FOUND != Result)
|
|
return FspFileSystemSendResponseWithStatus(FileSystem, Request, Result);
|
|
Create = TRUE;
|
|
}
|
|
|
|
if (!Create)
|
|
{
|
|
Result = FileSystem->Interface->FileOpen(FileSystem, Request, &FileNode);
|
|
if (!NT_SUCCESS(Result))
|
|
{
|
|
if (STATUS_OBJECT_NAME_NOT_FOUND != Result)
|
|
return FspFileSystemSendResponseWithStatus(FileSystem, Request, Result);
|
|
Create = TRUE;
|
|
}
|
|
}
|
|
|
|
if (Create)
|
|
{
|
|
if (FspIsRootDirectory((PWSTR)Request->Buffer))
|
|
return STATUS_ACCESS_DENIED;
|
|
|
|
Result = FspCreateCheck(FileSystem, Request, FALSE, &GrantedAccess);
|
|
if (!NT_SUCCESS(Result))
|
|
return FspFileSystemSendResponseWithStatus(FileSystem, Request, Result);
|
|
|
|
Result = FileSystem->Interface->FileCreate(FileSystem, Request, &FileNode);
|
|
if (!NT_SUCCESS(Result))
|
|
return FspFileSystemSendResponseWithStatus(FileSystem, Request, Result);
|
|
|
|
FileNode->Flags = 0;
|
|
memset(&FileNode->ShareAccess, 0, sizeof FileNode->ShareAccess);
|
|
}
|
|
|
|
Result = FspShareCheck(FileSystem, GrantedAccess, Request->Req.Create.ShareAccess, FileNode);
|
|
if (!NT_SUCCESS(Result))
|
|
{
|
|
if (0 != FileSystem->Interface->FileClose)
|
|
FileSystem->Interface->FileClose(FileSystem, Request, FileNode);
|
|
return FspFileSystemSendResponseWithStatus(FileSystem, Request, Result);
|
|
}
|
|
|
|
memset(&Response, 0, sizeof Response);
|
|
Response.Size = sizeof Response;
|
|
Response.Kind = Request->Kind;
|
|
Response.Hint = Request->Hint;
|
|
Response.IoStatus.Status = STATUS_SUCCESS;
|
|
Response.IoStatus.Information = Create ? FILE_CREATED : FILE_OPENED;
|
|
Response.Rsp.Create.Opened.UserContext = (UINT_PTR)FileNode;
|
|
Response.Rsp.Create.Opened.GrantedAccess = GrantedAccess;
|
|
return FspFileSystemSendResponse(FileSystem, &Response);
|
|
}
|
|
|
|
static NTSTATUS FspFileSystemOpCreate_FileOverwrite(FSP_FILE_SYSTEM *FileSystem,
|
|
FSP_FSCTL_TRANSACT_REQ *Request, BOOLEAN Supersede)
|
|
{
|
|
if (FspIsRootDirectory((PWSTR)Request->Buffer))
|
|
return STATUS_ACCESS_DENIED;
|
|
|
|
NTSTATUS Result;
|
|
DWORD GrantedAccess;
|
|
FSP_FILE_NODE *FileNode;
|
|
FSP_FSCTL_TRANSACT_RSP Response;
|
|
|
|
Result = FspAccessCheck(FileSystem, Request, TRUE,
|
|
Request->Req.Create.DesiredAccess | (Supersede ? DELETE : FILE_WRITE_DATA),
|
|
&GrantedAccess);
|
|
if (!NT_SUCCESS(Result))
|
|
return FspFileSystemSendResponseWithStatus(FileSystem, Request, Result);
|
|
|
|
if (0 == (Request->Req.Create.DesiredAccess & MAXIMUM_ALLOWED))
|
|
GrantedAccess &= Supersede ?
|
|
(~DELETE | (Request->Req.Create.DesiredAccess & DELETE)) :
|
|
(~FILE_WRITE_DATA | (Request->Req.Create.DesiredAccess & FILE_WRITE_DATA));
|
|
|
|
Result = FileSystem->Interface->FileOverwrite(FileSystem, Request, &FileNode);
|
|
if (!NT_SUCCESS(Result))
|
|
return FspFileSystemSendResponseWithStatus(FileSystem, Request, Result);
|
|
|
|
Result = FspShareCheck(FileSystem, GrantedAccess, Request->Req.Create.ShareAccess, FileNode);
|
|
if (!NT_SUCCESS(Result))
|
|
{
|
|
if (0 != FileSystem->Interface->FileClose)
|
|
FileSystem->Interface->FileClose(FileSystem, Request, FileNode);
|
|
return FspFileSystemSendResponseWithStatus(FileSystem, Request, Result);
|
|
}
|
|
|
|
memset(&Response, 0, sizeof Response);
|
|
Response.Size = sizeof Response;
|
|
Response.Kind = Request->Kind;
|
|
Response.Hint = Request->Hint;
|
|
Response.IoStatus.Status = STATUS_SUCCESS;
|
|
Response.IoStatus.Information = Supersede ? FILE_SUPERSEDED : FILE_OVERWRITTEN;
|
|
Response.Rsp.Create.Opened.UserContext = (UINT_PTR)FileNode;
|
|
Response.Rsp.Create.Opened.GrantedAccess = GrantedAccess;
|
|
return FspFileSystemSendResponse(FileSystem, &Response);
|
|
}
|
|
|
|
static NTSTATUS FspFileSystemOpCreate_FileOverwriteIf(FSP_FILE_SYSTEM *FileSystem,
|
|
FSP_FSCTL_TRANSACT_REQ *Request)
|
|
{
|
|
if (FspIsRootDirectory((PWSTR)Request->Buffer))
|
|
return STATUS_ACCESS_DENIED;
|
|
|
|
NTSTATUS Result;
|
|
DWORD GrantedAccess;
|
|
FSP_FILE_NODE *FileNode;
|
|
FSP_FSCTL_TRANSACT_RSP Response;
|
|
BOOLEAN Create = FALSE;
|
|
|
|
Result = FspAccessCheck(FileSystem, Request, TRUE,
|
|
Request->Req.Create.DesiredAccess | FILE_WRITE_DATA,
|
|
&GrantedAccess);
|
|
if (!NT_SUCCESS(Result))
|
|
{
|
|
if (STATUS_OBJECT_NAME_NOT_FOUND != Result)
|
|
return FspFileSystemSendResponseWithStatus(FileSystem, Request, Result);
|
|
Create = TRUE;
|
|
}
|
|
else
|
|
{
|
|
if (0 == (Request->Req.Create.DesiredAccess & MAXIMUM_ALLOWED))
|
|
GrantedAccess &= ~FILE_WRITE_DATA | (Request->Req.Create.DesiredAccess & FILE_WRITE_DATA);
|
|
}
|
|
|
|
if (!Create)
|
|
{
|
|
Result = FileSystem->Interface->FileOverwrite(FileSystem, Request, &FileNode);
|
|
if (!NT_SUCCESS(Result))
|
|
{
|
|
if (STATUS_OBJECT_NAME_NOT_FOUND != Result)
|
|
return FspFileSystemSendResponseWithStatus(FileSystem, Request, Result);
|
|
Create = TRUE;
|
|
}
|
|
}
|
|
|
|
if (Create)
|
|
{
|
|
Result = FspCreateCheck(FileSystem, Request, FALSE, &GrantedAccess);
|
|
if (!NT_SUCCESS(Result))
|
|
return FspFileSystemSendResponseWithStatus(FileSystem, Request, Result);
|
|
|
|
Result = FileSystem->Interface->FileCreate(FileSystem, Request, &FileNode);
|
|
if (!NT_SUCCESS(Result))
|
|
return FspFileSystemSendResponseWithStatus(FileSystem, Request, Result);
|
|
|
|
FileNode->Flags = 0;
|
|
memset(&FileNode->ShareAccess, 0, sizeof FileNode->ShareAccess);
|
|
}
|
|
|
|
Result = FspShareCheck(FileSystem, GrantedAccess, Request->Req.Create.ShareAccess, FileNode);
|
|
if (!NT_SUCCESS(Result))
|
|
{
|
|
if (0 != FileSystem->Interface->FileClose)
|
|
FileSystem->Interface->FileClose(FileSystem, Request, FileNode);
|
|
return FspFileSystemSendResponseWithStatus(FileSystem, Request, Result);
|
|
}
|
|
|
|
memset(&Response, 0, sizeof Response);
|
|
Response.Size = sizeof Response;
|
|
Response.Kind = Request->Kind;
|
|
Response.Hint = Request->Hint;
|
|
Response.IoStatus.Status = STATUS_SUCCESS;
|
|
Response.IoStatus.Information = Create ? FILE_CREATED : FILE_OVERWRITTEN;
|
|
Response.Rsp.Create.Opened.UserContext = (UINT_PTR)FileNode;
|
|
Response.Rsp.Create.Opened.GrantedAccess = GrantedAccess;
|
|
return FspFileSystemSendResponse(FileSystem, &Response);
|
|
}
|
|
|
|
static NTSTATUS FspFileSystemOpCreate_FileOpenTargetDirectory(FSP_FILE_SYSTEM *FileSystem,
|
|
FSP_FSCTL_TRANSACT_REQ *Request)
|
|
{
|
|
if (FspIsRootDirectory((PWSTR)Request->Buffer))
|
|
return STATUS_ACCESS_DENIED;
|
|
|
|
NTSTATUS Result;
|
|
DWORD GrantedAccess;
|
|
FSP_FILE_NODE *FileNode;
|
|
BOOLEAN FileExists;
|
|
FSP_FSCTL_TRANSACT_RSP Response;
|
|
PWSTR Path, Suffix;
|
|
|
|
FspPathSuffix((PWSTR)Request->Buffer, &Path, &Suffix);
|
|
Result = FspAccessCheck(FileSystem, Request, TRUE,
|
|
Request->Req.Create.DesiredAccess, &GrantedAccess);
|
|
FspPathCombine((PWSTR)Request->Buffer, Suffix);
|
|
|
|
if (!NT_SUCCESS(Result))
|
|
return FspFileSystemSendResponseWithStatus(FileSystem, Request, Result);
|
|
|
|
Result = FileSystem->Interface->FileOpenTargetDirectory(FileSystem, Request,
|
|
&FileNode, &FileExists);
|
|
if (!NT_SUCCESS(Result))
|
|
return FspFileSystemSendResponseWithStatus(FileSystem, Request, Result);
|
|
|
|
Result = FspShareCheck(FileSystem, GrantedAccess, Request->Req.Create.ShareAccess, FileNode);
|
|
if (!NT_SUCCESS(Result))
|
|
{
|
|
if (0 != FileSystem->Interface->FileClose)
|
|
FileSystem->Interface->FileClose(FileSystem, Request, FileNode);
|
|
return FspFileSystemSendResponseWithStatus(FileSystem, Request, Result);
|
|
}
|
|
|
|
memset(&Response, 0, sizeof Response);
|
|
Response.Size = sizeof Response;
|
|
Response.Kind = Request->Kind;
|
|
Response.Hint = Request->Hint;
|
|
Response.IoStatus.Status = STATUS_SUCCESS;
|
|
Response.IoStatus.Information = FileExists ? FILE_EXISTS : FILE_DOES_NOT_EXIST;
|
|
Response.Rsp.Create.Opened.UserContext = (UINT_PTR)FileNode;
|
|
Response.Rsp.Create.Opened.GrantedAccess = GrantedAccess;
|
|
return FspFileSystemSendResponse(FileSystem, &Response);
|
|
}
|
|
|
|
FSP_API NTSTATUS FspFileSystemOpCreate(FSP_FILE_SYSTEM *FileSystem,
|
|
FSP_FSCTL_TRANSACT_REQ *Request)
|
|
{
|
|
if (0 == FileSystem->Interface->FileCreate ||
|
|
0 == FileSystem->Interface->FileOpen ||
|
|
0 == FileSystem->Interface->FileOverwrite ||
|
|
0 == FileSystem->Interface->FileOpenTargetDirectory)
|
|
return FspFileSystemSendResponseWithStatus(FileSystem, Request, STATUS_INVALID_DEVICE_REQUEST);
|
|
|
|
if (Request->Req.Create.OpenTargetDirectory)
|
|
return FspFileSystemOpCreate_FileOpenTargetDirectory(FileSystem, Request);
|
|
|
|
switch ((Request->Req.Create.CreateOptions >> 24) & 0xff)
|
|
{
|
|
case FILE_CREATE:
|
|
return FspFileSystemOpCreate_FileCreate(FileSystem, Request);
|
|
case FILE_OPEN:
|
|
return FspFileSystemOpCreate_FileOpen(FileSystem, Request);
|
|
case FILE_OPEN_IF:
|
|
return FspFileSystemOpCreate_FileOpenIf(FileSystem, Request);
|
|
case FILE_OVERWRITE:
|
|
return FspFileSystemOpCreate_FileOverwrite(FileSystem, Request, FALSE);
|
|
case FILE_SUPERSEDE:
|
|
return FspFileSystemOpCreate_FileOverwrite(FileSystem, Request, TRUE);
|
|
case FILE_OVERWRITE_IF:
|
|
return FspFileSystemOpCreate_FileOverwriteIf(FileSystem, Request);
|
|
default:
|
|
return FspFileSystemSendResponseWithStatus(FileSystem, Request, STATUS_INVALID_PARAMETER);
|
|
}
|
|
}
|