1484 lines
35 KiB
C
1484 lines
35 KiB
C
/*++
|
||
|
||
Copyright (C) Microsoft Corporation, 1996 - 2001
|
||
(c) 1998 Seagate Software, Inc. All rights reserved.
|
||
|
||
Module Name:
|
||
|
||
RpFsa.h
|
||
|
||
Abstract:
|
||
|
||
Contains function declarations and structures for the File System Filter for Remote Storage
|
||
|
||
Author:
|
||
|
||
Rick Winter
|
||
|
||
Environment:
|
||
|
||
Kernel mode
|
||
|
||
|
||
Revision History:
|
||
|
||
X-13 108353 Michael C. Johnson 3-May-2001
|
||
When checking a file to determine the type of recall also
|
||
check a the potential target disk to see whether or not
|
||
it is writable. This is necessary now that we have read-only
|
||
NTFS volumes.
|
||
|
||
X-12 365077 Michael C. Johnson 1-May-2001
|
||
Revert to previous form of RsOpenTarget() with extra access
|
||
parameter to allow us to apply the desired access bypassing
|
||
the access check.
|
||
|
||
X-11 194325 Michael C. Johnson 1-Mar-2001
|
||
Clean up RsMountCompletion() and RsLoadFsCompletion() to
|
||
ensure they don't call routines such as IoDeleteDevice()
|
||
if not running at PASSIVE_LEVEL.
|
||
|
||
Add in memory trace mechanism in preparation for attempts
|
||
to flush out lingering reparse point deletion troubles.
|
||
|
||
|
||
X-10 326345 Michael C. Johnson 26-Feb-2001
|
||
Only send a single RP_RECALL_WAITING to the fsa on any one
|
||
file object. Use the new flag RP_NOTIFICATION_SENT to record
|
||
when notification has been done.
|
||
|
||
|
||
|
||
--*/
|
||
|
||
|
||
/* Defines */
|
||
|
||
// memory allocation Tags for debug usage
|
||
#define RP_RQ_TAG 'SFSR' // Recall queue
|
||
#define RP_FN_TAG 'NFSR' // File name cache
|
||
#define RP_SE_TAG 'ESSR' // Security info
|
||
#define RP_WQ_TAG 'QWSR' // Work queue
|
||
#define RP_QI_TAG 'IQSR' // Work Q info
|
||
#define RP_LT_TAG 'TLSR' // Long term memory
|
||
#define RP_IO_TAG 'OISR' // IOCTL queue
|
||
#define RP_FO_TAG 'OFSR' // File Object Queue
|
||
#define RP_VO_TAG 'OVSR' // Validate Queue
|
||
#define RP_ER_TAG 'RESR' // Error log data
|
||
#define RP_CC_TAG 'CCSR' // Cache buffers
|
||
#define RP_US_TAG 'SUSR' // Usn record
|
||
#define RP_CX_TAG 'CCSR' // Completion context
|
||
#define RP_TC_TAG 'CTSR' // Trace control block
|
||
#define RP_TE_TAG 'ETSR' // Trace entry buffer
|
||
#define RP_RD_TAG 'DRSR' // Root directory path
|
||
|
||
|
||
|
||
//
|
||
// Device extension for the RsFilter device object
|
||
//
|
||
typedef enum _RP_VOLUME_WRITE_STATUS {
|
||
RsVolumeStatusUnknown = 0, // No attempt has been made to determine volume writeability
|
||
// or attempt to determine volume writeability failed
|
||
RsVolumeStatusReadOnly, // volume is readonly
|
||
RsVolumeStatusReadWrite // Volume is writeable
|
||
} RP_VOLUME_WRITE_STATUS;
|
||
|
||
typedef struct _DEVICE_EXTENSION {
|
||
CSHORT Type;
|
||
CSHORT Size;
|
||
PDEVICE_OBJECT FileSystemDeviceObject;
|
||
|
||
PDEVICE_OBJECT RealDeviceObject;
|
||
|
||
BOOLEAN Attached;
|
||
|
||
BOOLEAN AttachedToNtfsControlDevice;
|
||
|
||
volatile RP_VOLUME_WRITE_STATUS WriteStatus;
|
||
|
||
} DEVICE_EXTENSION, *PDEVICE_EXTENSION;
|
||
|
||
|
||
#define RSFILTER_PARAMS_KEY L"RsFilter\\Parameters"
|
||
#define RS_TRACE_LEVEL_VALUE_NAME L"TraceLevel"
|
||
#define RS_TRACE_LEVEL_DEFAULT 0
|
||
|
||
|
||
extern PDEVICE_OBJECT FsDeviceObject;
|
||
|
||
|
||
// Fsa validate job registry entry location
|
||
|
||
#define FSA_VALIDATE_LOG_KEY_NAME L"\\Registry\\Machine\\SYSTEM\\CurrentControlSet\\Services\\Remote_Storage_File_System_Agent\\Validate"
|
||
|
||
#define FT_VOL_LEN 32
|
||
|
||
/* First guess at device name length */
|
||
#define AV_DEV_OBJ_NAME_SIZE (40 * sizeof(wchar_t))
|
||
|
||
/* Space for a NULL and a delimiter */
|
||
#define AV_NAME_OVERHEAD (2 * sizeof(wchar_t))
|
||
|
||
#define RP_NTFS_NAME L"\\FileSystem\\NTFS"
|
||
|
||
// FILE_HSM_ACTION_ACCESS is any access that requires HSM action (delete or recall or both)
|
||
|
||
#ifdef WHEN_WE_HANDLE_DELETE
|
||
#define FILE_HSM_ACTION_ACCESS (FILE_READ_DATA | FILE_WRITE_DATA | FILE_EXECUTE | DELETE)
|
||
#else
|
||
#define FILE_HSM_ACTION_ACCESS (FILE_READ_DATA | FILE_WRITE_DATA | FILE_EXECUTE)
|
||
#endif
|
||
|
||
/* FILE_HSM_RECALL_ACCESS is any access that allows the data to be read. */
|
||
|
||
#define FILE_HSM_RECALL_ACCESS (FILE_READ_DATA | FILE_WRITE_DATA | FILE_EXECUTE)
|
||
|
||
//
|
||
// Timeout and retry values when waiting for the FSA to issue an IOCTL
|
||
// Represents the amount of time - under multiple concurrent recall situations -
|
||
// that an app will have to wait before the i/o it issued completes with
|
||
// STATUS_FILE_IS_OFFLINE because RsFilter couldn't get any IOCTLs to
|
||
// communicate with the FSA
|
||
//
|
||
#define RP_WAIT_FOR_FSA_IO_TIMEOUT -((LONGLONG) 4800000000) // 8 minutes
|
||
|
||
/* Module ID defines for error/event logging */
|
||
|
||
#define AV_MODULE_RPFILTER 1
|
||
#define AV_MODULE_RPFILFUN 2
|
||
#define AV_MODULE_RPSEC 3
|
||
#define AV_MODULE_RPZW 4
|
||
#define AV_MODULE_RPCACHE 5
|
||
|
||
#define AV_BUFFER_SIZE 1024
|
||
|
||
|
||
#ifndef BooleanFlagOn
|
||
#define BooleanFlagOn(F,SF) ( (BOOLEAN)(((F) & (SF)) != 0) )
|
||
#endif
|
||
|
||
|
||
#define AV_FT_TICKS_PER_SECOND ((LONGLONG) 10000000)
|
||
#define AV_FT_TICKS_PER_MINUTE ((LONGLONG) ((LONGLONG) 60 * AV_FT_TICKS_PER_SECOND))
|
||
#define AV_FT_TICKS_PER_HOUR ((LONGLONG) ((LONGLONG) 60 * AV_FT_TICKS_PER_MINUTE))
|
||
|
||
|
||
//
|
||
// The filter ID tracks recalls and no-recalls as follows:
|
||
// The id is a longlong where the highest order bit identifies the type of recall
|
||
// (no-recall or recall). The remaining part of the high order long identifies the
|
||
// read RP_IRP_QUEUE entry (for no-recall) or the file object entry (for recall).
|
||
// The lower long identifies the file context entry.
|
||
//
|
||
|
||
#define RP_TYPE_RECALL (ULONGLONG) 0x8000000000000000
|
||
#define RP_CONTEXT_MASK (ULONGLONG) 0x00000000ffffffff
|
||
#define RP_READ_MASK 0x7fffffff
|
||
#define RP_FILE_MASK (ULONGLONG) 0xffffffff00000000
|
||
|
||
typedef struct _RP_CREATE_INFO {
|
||
PIRP irp;
|
||
PIO_STACK_LOCATION irpSp;
|
||
POBJECT_NAME_INFORMATION str;
|
||
ULONG options;
|
||
//
|
||
// Reparse point data
|
||
//
|
||
RP_DATA rpData;
|
||
LONGLONG fileId;
|
||
LONGLONG objIdHi;
|
||
LONGLONG objIdLo;
|
||
ULONG serial;
|
||
ULONG action;
|
||
ULONG desiredAccess;
|
||
} RP_CREATE_INFO, *PRP_CREATE_INFO;
|
||
|
||
|
||
typedef struct _RP_PENDING_CREATE {
|
||
//
|
||
// Filter id
|
||
//
|
||
ULONGLONG filterId;
|
||
//
|
||
//
|
||
//
|
||
PRP_CREATE_INFO qInfo;
|
||
//
|
||
// Event used to signal irp completion
|
||
//
|
||
KEVENT irpCompleteEvent;
|
||
//
|
||
// File object for irp
|
||
//
|
||
PFILE_OBJECT fileObject;
|
||
//
|
||
// Device object for irp
|
||
//
|
||
PDEVICE_OBJECT deviceObject;
|
||
//
|
||
// Open options
|
||
//
|
||
ULONG options;
|
||
|
||
|
||
//
|
||
// Indicates if oplocks should not be granted (to CI for instance..)
|
||
//
|
||
#define RP_PENDING_NO_OPLOCK 0x1
|
||
//
|
||
// Indicates if IRP should be sent down again
|
||
//
|
||
#define RP_PENDING_RESEND_IRP 0x2
|
||
//
|
||
// Indicates if we should wait for irp to complete
|
||
//
|
||
#define RP_PENDING_WAIT_FOR_EVENT 0x4
|
||
//
|
||
// Indicates if this is a recall
|
||
//
|
||
#define RP_PENDING_IS_RECALL 0x8
|
||
//
|
||
// Indicates if we should reset the offline attribute of the file
|
||
//
|
||
#define RP_PENDING_RESET_OFFLINE 0x10
|
||
ULONG flags;
|
||
} RP_PENDING_CREATE, *PRP_PENDING_CREATE;
|
||
|
||
|
||
#define RP_IRP_NO_RECALL 1
|
||
|
||
typedef struct _RP_IRP_QUEUE {
|
||
LIST_ENTRY list;
|
||
PIRP irp;
|
||
PDEVICE_EXTENSION deviceExtension;
|
||
ULONG flags;
|
||
//
|
||
// For regular read and write, offset and length
|
||
// denote the offset and length within the file
|
||
// For no-recall reads, offset and length would
|
||
// denote the offset/length within the cacheBuffer
|
||
//
|
||
ULONGLONG offset;
|
||
ULONGLONG length;
|
||
//
|
||
// These fields are used only for no-recall reads
|
||
// filterId for no-recall (see filterid description)
|
||
ULONGLONG readId;
|
||
ULONGLONG recallOffset;
|
||
ULONGLONG recallLength;
|
||
//
|
||
// User buffer for data from read-no-recall
|
||
//
|
||
PVOID userBuffer;
|
||
//
|
||
// Cache block buffer for no recall data
|
||
//
|
||
PVOID cacheBuffer;
|
||
} RP_IRP_QUEUE, *PRP_IRP_QUEUE;
|
||
|
||
//
|
||
// Structure tracking the no-recall master IRP and associated irps
|
||
//
|
||
typedef struct _RP_NO_RECALL_MASTER_IRP {
|
||
LIST_ENTRY AssocIrps;
|
||
PIRP MasterIrp;
|
||
} RP_NO_RECALL_MASTER_IRP, *PRP_NO_RECALL_MASTER_IRP;
|
||
|
||
//
|
||
// User security info structure: this is required for HSM
|
||
// to do the pop-up for clients indicating the file is being recalled
|
||
//
|
||
typedef struct _RP_USER_SECURITY_INFO {
|
||
//
|
||
// Sid info
|
||
//
|
||
PCHAR userInfo;
|
||
ULONG userInfoLen;
|
||
LUID userAuthentication;
|
||
LUID userInstance;
|
||
LUID tokenSourceId;
|
||
//
|
||
// Token source info for user
|
||
//
|
||
CHAR tokenSource[TOKEN_SOURCE_LENGTH];
|
||
//
|
||
// Indicates if this was opened by user with admin privileges
|
||
//
|
||
BOOLEAN isAdmin;
|
||
//
|
||
// Indicates if this is a local proc
|
||
//
|
||
BOOLEAN localProc;
|
||
|
||
} RP_USER_SECURITY_INFO, *PRP_USER_SECURITY_INFO;
|
||
|
||
//
|
||
// Associated macro for above
|
||
//
|
||
#define RsFreeUserSecurityInfo(UserSecurityInfo) { \
|
||
if (UserSecurityInfo) { \
|
||
if (UserSecurityInfo->userInfo) { \
|
||
ExFreePool(UserSecurityInfo->userInfo); \
|
||
} \
|
||
ExFreePool(UserSecurityInfo); \
|
||
} \
|
||
}
|
||
|
||
//
|
||
// The file object entry keeps track of an open instance of a file.
|
||
// For each NTFS file object there is one of these (if the file has an HSM tag)
|
||
// This structure points to a FS_CONTEXT entry for which there is one for each file.
|
||
// For instance if 3 clients open \\server\share\foo there will be 3 file object
|
||
// structures and they will all point to the same FS_CONTEXT structure.
|
||
//
|
||
// The file objects we are tracking will have a pointer to one of there structures attached via
|
||
// FsRtlInsertFilterContext. From there one can find the file context entry via the pointer to it.
|
||
//
|
||
typedef struct _RP_FILE_OBJ {
|
||
//
|
||
// Link to next file object
|
||
//
|
||
LIST_ENTRY list;
|
||
//
|
||
// File object itself
|
||
//
|
||
PFILE_OBJECT fileObj;
|
||
//
|
||
// Device object
|
||
//
|
||
PDEVICE_OBJECT devObj;
|
||
//
|
||
// Pointer to the RP_FILE_CONTEXT entry - there's one such entry for every *file*
|
||
//
|
||
PVOID fsContext;
|
||
//
|
||
// Resource protecting this entry
|
||
//
|
||
ERESOURCE resource;
|
||
//
|
||
// Spin lock protecting read/write IRP queues
|
||
//
|
||
KSPIN_LOCK qLock;
|
||
//
|
||
// Pending read IRP queue
|
||
//
|
||
LIST_ENTRY readQueue;
|
||
//
|
||
// Pending write IRP queue
|
||
//
|
||
LIST_ENTRY writeQueue;
|
||
//
|
||
// File create options specified when opening it
|
||
//
|
||
ULONG openOptions;
|
||
//
|
||
// File desired access spcecified when opening it
|
||
//
|
||
ULONG desiredAccess;
|
||
//
|
||
// Flags (descriptions below)
|
||
//
|
||
ULONG flags;
|
||
//
|
||
// Object id
|
||
//
|
||
LONGLONG objIdHi;
|
||
LONGLONG objIdLo;
|
||
//
|
||
// File Id if available
|
||
//
|
||
LONGLONG fileId;
|
||
//
|
||
// Unique ID we generate for the file object
|
||
//
|
||
ULONGLONG filterId;
|
||
//
|
||
// Recall action flags (see rpio.h - RP_RECALL_ACTION..)
|
||
//
|
||
ULONG recallAction;
|
||
PRP_USER_SECURITY_INFO userSecurityInfo;
|
||
} RP_FILE_OBJ, *PRP_FILE_OBJ;
|
||
|
||
//
|
||
// RP_FILE_OBJ Flags
|
||
//
|
||
//
|
||
// File was not opened for read or write access
|
||
//
|
||
#define RP_NO_DATA_ACCESS 1
|
||
//
|
||
// Opener is admin equivalent
|
||
//
|
||
#define RP_OPEN_BY_ADMIN 2
|
||
//
|
||
// Opened by local process
|
||
//
|
||
#define RP_OPEN_LOCAL 4
|
||
//
|
||
// Recall waiting notification already sent
|
||
//
|
||
#define RP_NOTIFICATION_SENT 8
|
||
|
||
|
||
//
|
||
// Recall state
|
||
//
|
||
typedef enum _RP_RECALL_STATE {
|
||
RP_RECALL_INVALID = -1,
|
||
RP_RECALL_NOT_RECALLED,
|
||
RP_RECALL_STARTED,
|
||
RP_RECALL_COMPLETED
|
||
} RP_RECALL_STATE, *PRP_RECALL_STATE;
|
||
|
||
//
|
||
// Filter context for RsFilter:
|
||
// Since filter contexts are attached to the SCB (stream control block) we need to use
|
||
// the instance ID to indicate which file object we are interested in. We attach this
|
||
// structure to and use myFileObjEntry to point to the RP_FILE_OBJ struct that
|
||
// represents this file object.
|
||
//
|
||
typedef struct _RP_FILTER_CONTEXT {
|
||
FSRTL_PER_STREAM_CONTEXT context;
|
||
PVOID myFileObjEntry;
|
||
} RP_FILTER_CONTEXT, *PRP_FILTER_CONTEXT;
|
||
|
||
//
|
||
// File context: one per *file*
|
||
//
|
||
typedef struct _RP_FILE_CONTEXT {
|
||
//
|
||
// Links to next/prev file (hanging off RsFileObjQHead)
|
||
//
|
||
LIST_ENTRY list;
|
||
//
|
||
// Lock protecting file object queue
|
||
//
|
||
KSPIN_LOCK qLock;
|
||
//
|
||
// Queue of all related file object entries
|
||
//
|
||
LIST_ENTRY fileObjects;
|
||
//
|
||
// Recalled data is written using this file object
|
||
//
|
||
PFILE_OBJECT fileObjectToWrite;
|
||
//
|
||
// Handle for the file object we use to write to
|
||
//
|
||
HANDLE handle;
|
||
|
||
PDEVICE_OBJECT devObj;
|
||
|
||
PDEVICE_OBJECT FilterDeviceObject;
|
||
|
||
//
|
||
// Unicode name of file
|
||
//
|
||
POBJECT_NAME_INFORMATION uniName;
|
||
//
|
||
// From the file object - unique file identifier
|
||
//
|
||
PVOID fsContext;
|
||
//
|
||
// Buffer to write out to file
|
||
//
|
||
PVOID nextWriteBuffer;
|
||
//
|
||
// Size of next write to the file (of recall data)
|
||
//
|
||
ULONG nextWriteSize;
|
||
//
|
||
// Lock protecting this entry
|
||
//
|
||
ERESOURCE resource;
|
||
//
|
||
// This notification event is signalled when recall completes for this file
|
||
//
|
||
KEVENT recallCompletedEvent;
|
||
//
|
||
// File id if available
|
||
//
|
||
LONGLONG fileId;
|
||
//
|
||
// Size in bytes of recall needed
|
||
//
|
||
LARGE_INTEGER recallSize;
|
||
//
|
||
// All bytes up to this offset have been recalled
|
||
//
|
||
LARGE_INTEGER currentOffset;
|
||
//
|
||
// Lower half of filter id (unique per file)
|
||
//
|
||
ULONGLONG filterId;
|
||
//
|
||
// filterId for last completion
|
||
//
|
||
ULONGLONG lastRecallCompletionFilterId;
|
||
//
|
||
// Volume serial number
|
||
//
|
||
ULONG serial;
|
||
//
|
||
// If the recall is complete this is the status
|
||
//
|
||
NTSTATUS recallStatus;
|
||
//
|
||
// Recall state
|
||
//
|
||
RP_RECALL_STATE state;
|
||
//
|
||
// Flags (see below for description)
|
||
//
|
||
ULONG flags;
|
||
//
|
||
// Reference count for the file context
|
||
//
|
||
ULONG refCount;
|
||
//
|
||
// Usn of the file
|
||
//
|
||
USN usn;
|
||
//
|
||
// Tracks create section lock
|
||
//
|
||
LONG createSectionLock;
|
||
//
|
||
// Reparse point data
|
||
//
|
||
RP_DATA rpData;
|
||
|
||
} RP_FILE_CONTEXT, *PRP_FILE_CONTEXT;
|
||
|
||
//
|
||
// RP_FILE_CONTEXT Flags
|
||
//
|
||
// We have seen a write to this file
|
||
#define RP_FILE_WAS_WRITTEN 1
|
||
#define RP_FILE_INITIALIZED 2
|
||
#define RP_FILE_REPARSE_POINT_DELETED 4
|
||
|
||
/*++
|
||
|
||
VOID
|
||
RsInitializeFileContextQueueLock()
|
||
|
||
Routine Description
|
||
|
||
Initializes lock guarding the file context queue
|
||
|
||
Arguments
|
||
|
||
none
|
||
|
||
Return Value
|
||
|
||
none
|
||
|
||
--*/
|
||
#define RsInitializeFileContextQueueLock() { \
|
||
DebugTrace((DPFLTR_RSFILTER_ID, DBG_LOCK,"RsFilter: RsInitializeFileContextQueueLock.\n"));\
|
||
ExInitializeFastMutex(&RsFileContextQueueLock); \
|
||
}
|
||
|
||
/*++
|
||
|
||
VOID
|
||
RsAcquireFileContextQueueLock()
|
||
|
||
Routine Description
|
||
|
||
Acquire lock guarding the file context queue
|
||
|
||
Arguments
|
||
|
||
none
|
||
|
||
Return Value
|
||
|
||
none
|
||
|
||
--*/
|
||
#define RsAcquireFileContextQueueLock() { \
|
||
ExAcquireFastMutex(&RsFileContextQueueLock); \
|
||
DebugTrace((DPFLTR_RSFILTER_ID, DBG_LOCK, "RsFilter: RsAcquireFileContextQueueLock.\n"));\
|
||
}
|
||
|
||
/*++
|
||
|
||
VOID
|
||
RsReleaseFileContextQueueLock()
|
||
|
||
Routine Description
|
||
|
||
Release lock guarding the file context queue
|
||
|
||
Arguments
|
||
|
||
none
|
||
|
||
Return Value
|
||
|
||
none
|
||
|
||
--*/
|
||
#define RsReleaseFileContextQueueLock() { \
|
||
DebugTrace((DPFLTR_RSFILTER_ID,DBG_LOCK, "RsFilter: RsReleaseFileContextQueueLock.\n"));\
|
||
ExReleaseFastMutex(&RsFileContextQueueLock); \
|
||
}
|
||
|
||
|
||
|
||
/*++
|
||
|
||
VOID
|
||
RsAcquireFileObjectLockExclusive()
|
||
|
||
Routine Description
|
||
|
||
Acquire lock guarding a file object entry
|
||
|
||
Arguments
|
||
|
||
none
|
||
|
||
Return Value
|
||
|
||
none
|
||
|
||
--*/
|
||
#define RsAcquireFileObjectEntryLockExclusive(entry) { \
|
||
DebugTrace((DPFLTR_RSFILTER_ID,DBG_LOCK, "RsFilter: RsAcquireFileObjectEntryLockExclusive Waiting (%x).\n", entry));\
|
||
FsRtlEnterFileSystem(); \
|
||
ExAcquireResourceExclusiveLite(&(entry)->resource, TRUE); \
|
||
DebugTrace((DPFLTR_RSFILTER_ID,DBG_LOCK, "RsFilter: RsAcquireFileObjectEntryLockExclusive Owned (%x).\n", entry));\
|
||
}
|
||
|
||
/*++
|
||
VOID
|
||
RsAcquireFileObjectEntryLockShared()
|
||
|
||
Routine Description
|
||
|
||
Acquire lock guarding a file object entry
|
||
|
||
Arguments
|
||
|
||
none
|
||
|
||
Return Value
|
||
|
||
none
|
||
|
||
--*/
|
||
#define RsAcquireFileObjectEntryLockShared(entry) { \
|
||
DebugTrace((DPFLTR_RSFILTER_ID,DBG_LOCK, "RsFilter: RsAcquireFileObjectEntryLockShared Waiting (%x).\n", entry));\
|
||
FsRtlEnterFileSystem(); \
|
||
ExAcquireResourceSharedLite(&(entry)->resource, TRUE); \
|
||
DebugTrace((DPFLTR_RSFILTER_ID,DBG_LOCK, "RsFilter: RsAcquireFileObjectEntryLockShared Owned (%x).\n", entry));\
|
||
}
|
||
|
||
/*++
|
||
|
||
VOID
|
||
RsReleaseFileObjectEntryLock()
|
||
|
||
Routine Description
|
||
|
||
Release lock guarding a file object entry
|
||
|
||
Arguments
|
||
|
||
none
|
||
|
||
Return Value
|
||
|
||
none
|
||
|
||
--*/
|
||
#define RsReleaseFileObjectEntryLock(entry) { \
|
||
DebugTrace((DPFLTR_RSFILTER_ID,DBG_LOCK, "RsFilter: RsReleaseFileObjectEntryLock (%x).\n", entry));\
|
||
ExReleaseResourceLite(&(entry)->resource); \
|
||
FsRtlExitFileSystem(); \
|
||
}
|
||
|
||
|
||
/*++
|
||
|
||
VOID
|
||
RsAcquireFileContextEntryLockExclusive()
|
||
|
||
Routine Description
|
||
|
||
Acquire lock guarding a file context entry
|
||
|
||
Arguments
|
||
|
||
none
|
||
|
||
Return Value
|
||
|
||
none
|
||
|
||
--*/
|
||
#define RsAcquireFileContextEntryLockExclusive(entry) { \
|
||
DebugTrace((DPFLTR_RSFILTER_ID,DBG_LOCK, "RsFilter: RsAcquireFileContextEntryLockExclusive Waiting (%x).\n", entry));\
|
||
FsRtlEnterFileSystem(); \
|
||
ExAcquireResourceExclusiveLite(&(entry)->resource, TRUE); \
|
||
DebugTrace((DPFLTR_RSFILTER_ID,DBG_LOCK, "RsFilter: RsAcquireFileContextEntryLockExclusive Owned (%x).\n", entry));\
|
||
}
|
||
|
||
/*++
|
||
|
||
VOID
|
||
RsAcquireFileContextEntryLockShared()
|
||
|
||
Routine Description
|
||
|
||
Acquire lock guarding a file context entry
|
||
|
||
Arguments
|
||
|
||
none
|
||
|
||
Return Value
|
||
|
||
none
|
||
|
||
--*/
|
||
#define RsAcquireFileContextEntryLockShared(entry) { \
|
||
DebugTrace((DPFLTR_RSFILTER_ID,DBG_LOCK, "RsFilter: RsAcquireFileContextEntryLockShared Waiting (%x).\n", entry));\
|
||
FsRtlEnterFileSystem(); \
|
||
ExAcquireResourceSharedLite(&(entry)->resource, TRUE); \
|
||
DebugTrace((DPFLTR_RSFILTER_ID,DBG_LOCK, "RsFilter: RsAcquireFileContextEntryLockShared Owned (%x).\n", entry));\
|
||
}
|
||
|
||
/*++
|
||
|
||
VOID
|
||
RsReleaseFileContextEntryLock()
|
||
|
||
Routine Description
|
||
|
||
Release lock guarding a file context entry
|
||
|
||
Arguments
|
||
|
||
none
|
||
|
||
Return Value
|
||
|
||
none
|
||
|
||
--*/
|
||
#define RsReleaseFileContextEntryLock(entry) { \
|
||
DebugTrace((DPFLTR_RSFILTER_ID,DBG_LOCK, "RsFilter: RsReleaseFileContextEntryLock. (%x)\n", entry));\
|
||
ExReleaseResourceLite(&(entry)->resource); \
|
||
FsRtlExitFileSystem(); \
|
||
}
|
||
|
||
/*++
|
||
|
||
VOID
|
||
RsGetValidateLock(PKIRQL irql)
|
||
|
||
Routine Description:
|
||
|
||
Get a lock on the validate queue
|
||
|
||
Arguments:
|
||
Place to save irql
|
||
|
||
Return Value:
|
||
None
|
||
|
||
--*/
|
||
#define RsGetValidateLock(irql) ExAcquireSpinLock(&RsValidateQueueLock, irql)
|
||
|
||
/*++
|
||
|
||
VOID
|
||
RsPutValidateLock(KIRQL oldIrql)
|
||
|
||
Routine Description:
|
||
|
||
Free a lock on the validate queue
|
||
|
||
Arguments:
|
||
Saved irql
|
||
|
||
Return Value:
|
||
None
|
||
|
||
--*/
|
||
|
||
#define RsPutValidateLock(oldIrql) ExReleaseSpinLock(&RsValidateQueueLock, oldIrql)
|
||
|
||
/*++
|
||
|
||
VOID
|
||
RsGetIoLock(PKIRQL irql)
|
||
|
||
Routine Description:
|
||
|
||
Lock the IO queue
|
||
|
||
Arguments:
|
||
|
||
Variable to receive current irql
|
||
|
||
Return Value:
|
||
|
||
0
|
||
|
||
Note:
|
||
|
||
--*/
|
||
|
||
#define RsGetIoLock(irql) ExAcquireSpinLock(&RsIoQueueLock, irql)
|
||
|
||
/*++
|
||
|
||
VOID
|
||
RsPutIoLock(KIRQL oldIrql)
|
||
|
||
Routine Description:
|
||
|
||
Unlock the IO queue
|
||
|
||
Arguments:
|
||
|
||
oldIrql - Saved irql
|
||
|
||
Return Value:
|
||
|
||
0
|
||
|
||
Note:
|
||
|
||
--*/
|
||
|
||
#define RsPutIoLock(oldIrql) ExReleaseSpinLock(&RsIoQueueLock, oldIrql)
|
||
|
||
#define RP_IS_NO_RECALL_OPTION(OpenOptions) \
|
||
(RsNoRecallDefault?!((OpenOptions) & FILE_OPEN_NO_RECALL) : ((OpenOptions) & FILE_OPEN_NO_RECALL))
|
||
|
||
#define RP_SET_NO_RECALL_OPTION(OpenOptions) \
|
||
(RsNoRecallDefault ? ((OpenOptions) &= ~FILE_OPEN_NO_RECALL):((OpenOptions) |= FILE_OPEN_NO_RECALL))
|
||
|
||
#define RP_RESET_NO_RECALL_OPTION(OpenOptions) \
|
||
(RsNoRecallDefault ?((OpenOptions) |= FILE_OPEN_NO_RECALL) : ((OpenOptions) &= ~FILE_OPEN_NO_RECALL))
|
||
|
||
|
||
#define RP_IS_NO_RECALL(Entry) \
|
||
(RP_IS_NO_RECALL_OPTION((Entry)->openOptions) && !(((PRP_FILE_CONTEXT) (Entry)->fsContext)->flags & RP_FILE_WAS_WRITTEN))
|
||
|
||
#define RP_SET_NO_RECALL(Entry) \
|
||
RP_SET_NO_RECALL_OPTION((Entry)->openOptions)
|
||
|
||
#define RP_RESET_NO_RECALL(Entry) \
|
||
RP_RESET_NO_RECALL_OPTION(Entry->openOptions)
|
||
|
||
|
||
typedef struct _RP_VALIDATE_INFO {
|
||
LIST_ENTRY list;
|
||
LARGE_INTEGER lastSetTime; // Last time a RP was set.
|
||
ULONG serial; // Volume serial number
|
||
} RP_VALIDATE_INFO, *PRP_VALIDATE_INFO;
|
||
|
||
typedef struct _AV_ERR {
|
||
ULONG line;
|
||
ULONG file;
|
||
ULONG code;
|
||
WCHAR string[1]; /* Actual size will vary */
|
||
} AV_ERR, *PAV_ERR;
|
||
|
||
//
|
||
// Possible create flags:
|
||
//
|
||
#define SF_FILE_CREATE_PATH 1
|
||
#define SF_FILE_CREATE_ID 2
|
||
#define SF_FILE_READ 3
|
||
|
||
|
||
typedef enum _RP_FILE_BUF_STATE {
|
||
RP_FILE_BUF_INVALID=0,
|
||
RP_FILE_BUF_IO,
|
||
RP_FILE_BUF_VALID,
|
||
RP_FILE_BUF_ERROR
|
||
} RP_FILE_BUF_STATE, *PRP_FILE_BUF_STATE;
|
||
|
||
//
|
||
// Define the cache buffer structure
|
||
//
|
||
typedef struct _RP_FILE_BUF {
|
||
//
|
||
// IRPs waiting on this block
|
||
//
|
||
LIST_ENTRY WaitQueue;
|
||
//
|
||
// Volume serial number for the volume on which the file
|
||
// this block maps to resides
|
||
//
|
||
ULONG VolumeSerial;
|
||
//
|
||
// File id uniquely indicating which file this block
|
||
// belongs to
|
||
//
|
||
ULONGLONG FileId;
|
||
//
|
||
// Block number this buffer maps to
|
||
//
|
||
ULONGLONG Block;
|
||
//
|
||
// Lock for the buffer
|
||
//
|
||
ERESOURCE Lock;
|
||
//
|
||
// Links in the hash queue this buffer belongs
|
||
//
|
||
LIST_ENTRY BucketLinks;
|
||
//
|
||
// Links in the lru list
|
||
//
|
||
LIST_ENTRY LruLinks;
|
||
//
|
||
// Indicates the current buffer state
|
||
//
|
||
RP_FILE_BUF_STATE State;
|
||
//
|
||
// If i/o completed with errors, this is useful
|
||
//
|
||
NTSTATUS IoStatus;
|
||
//
|
||
// Actual buffer contents themselves
|
||
//
|
||
PUCHAR Data;
|
||
//
|
||
// Usn used to validate block
|
||
//
|
||
LONGLONG Usn;
|
||
|
||
} RP_FILE_BUF, *PRP_FILE_BUF;
|
||
|
||
//
|
||
// The hash bucket structure
|
||
//
|
||
typedef struct _RP_CACHE_BUCKET {
|
||
//
|
||
// Link to the head of the entries in this bucket
|
||
//
|
||
LIST_ENTRY FileBufHead;
|
||
|
||
} RP_CACHE_BUCKET, *PRP_CACHE_BUCKET;
|
||
|
||
//
|
||
// Cache LRU structure
|
||
//
|
||
typedef struct _RP_CACHE_LRU {
|
||
//
|
||
// Pointer to head of LRU
|
||
//
|
||
LIST_ENTRY FileBufHead;
|
||
//
|
||
// Lock structure for protecting the LRU
|
||
//
|
||
FAST_MUTEX Lock;
|
||
//
|
||
// Total number of buffers in the cache
|
||
//
|
||
//
|
||
ULONG TotalCount;
|
||
//
|
||
// Number of buffers in LRU (just for bookkeeping)
|
||
//
|
||
ULONG LruCount;
|
||
//
|
||
// Counting semaphore used to signal availability (and number)
|
||
// of buffers in LRU
|
||
//
|
||
KSEMAPHORE AvailableSemaphore;
|
||
|
||
} RP_CACHE_LRU, *PRP_CACHE_LRU;
|
||
|
||
|
||
|
||
|
||
//
|
||
// Completion Context used by Mount and LoadFs completion routines.
|
||
//
|
||
|
||
typedef struct _RP_COMPLETION_CONTEXT {
|
||
LIST_ENTRY leQueueHead;
|
||
PIO_WORKITEM pIoWorkItem;
|
||
PIRP pIrp;
|
||
PIO_WORKITEM_ROUTINE prtnWorkItemRoutine;
|
||
union {
|
||
struct {
|
||
PVPB pvpbOriginalVpb;
|
||
PDEVICE_OBJECT pdoRealDevice;
|
||
PDEVICE_OBJECT pdoNewFilterDevice;
|
||
} Mount;
|
||
|
||
struct {
|
||
PVOID pvDummy;
|
||
} LoadFs;
|
||
} Parameters;
|
||
} RP_COMPLETION_CONTEXT, *PRP_COMPLETION_CONTEXT;
|
||
|
||
|
||
//
|
||
// Some utility macros
|
||
//
|
||
|
||
#define MAX(a,b) ((a) > (b) ? (a) : (b))
|
||
#define MIN(a,b) ((a) > (b) ? (b) : (a))
|
||
|
||
//
|
||
// Debug support
|
||
//
|
||
#define DBG_INFO DPFLTR_INFO_LEVEL
|
||
#define DBG_ERROR DPFLTR_ERROR_LEVEL
|
||
#define DBG_VERBOSE DPFLTR_TRACE_LEVEL
|
||
#define DBG_LOCK DPFLTR_TRACE_LEVEL
|
||
|
||
#define DebugTrace(MSG) DbgPrintEx MSG
|
||
|
||
|
||
//
|
||
// Function prototypes
|
||
//
|
||
|
||
|
||
NTSTATUS
|
||
RsAddQueue(IN ULONG Serial,
|
||
OUT PULONGLONG RecallId,
|
||
IN ULONG OpenOption,
|
||
IN PFILE_OBJECT FileObject,
|
||
IN PDEVICE_OBJECT DevObj,
|
||
IN PDEVICE_OBJECT FilterDeviceObject,
|
||
IN PRP_DATA PhData,
|
||
IN LARGE_INTEGER RecallStart,
|
||
IN LARGE_INTEGER RecallSize,
|
||
IN LONGLONG FileId,
|
||
IN LONGLONG ObjIdHi,
|
||
IN LONGLONG ObjIdLo,
|
||
IN ULONG DesiredAccess,
|
||
IN PRP_USER_SECURITY_INFO UserSecurityInfo);
|
||
|
||
NTSTATUS
|
||
RsAddFileObj(IN PFILE_OBJECT fileObj,
|
||
IN PDEVICE_OBJECT FilterDeviceObject,
|
||
IN RP_DATA *phData,
|
||
IN ULONG openOption);
|
||
|
||
|
||
NTSTATUS
|
||
RsQueueCancel(IN ULONGLONG filterId);
|
||
|
||
NTSTATUS
|
||
RsMakeContext(IN PFILE_OBJECT fileObj,
|
||
OUT PRP_FILE_CONTEXT *context);
|
||
|
||
NTSTATUS
|
||
RsReleaseFileContext(IN PRP_FILE_CONTEXT context);
|
||
|
||
NTSTATUS
|
||
RsFreeFileObject(IN PLIST_ENTRY FilterContext);
|
||
|
||
PRP_FILE_CONTEXT
|
||
RsAcquireFileContext(IN ULONGLONG FilterId,
|
||
IN BOOLEAN Exclusive);
|
||
|
||
VOID
|
||
RsReleaseFileObject(IN PRP_FILE_OBJ entry);
|
||
|
||
NTSTATUS
|
||
RsGenerateDevicePath(IN PDEVICE_OBJECT deviceObject,
|
||
OUT POBJECT_NAME_INFORMATION *nameInfo
|
||
);
|
||
|
||
NTSTATUS
|
||
RsGenerateFullPath(IN POBJECT_NAME_INFORMATION fileName,
|
||
IN PDEVICE_OBJECT deviceObject,
|
||
OUT POBJECT_NAME_INFORMATION *nameInfo
|
||
);
|
||
|
||
ULONG
|
||
RsRemoveQueue(IN PFILE_OBJECT fileObj);
|
||
|
||
NTSTATUS
|
||
RsCompleteRecall(IN PDEVICE_OBJECT DeviceObject,
|
||
IN ULONGLONG FilterId,
|
||
IN NTSTATUS Status,
|
||
IN ULONG RecallAction,
|
||
IN BOOLEAN CancellableRead);
|
||
|
||
NTSTATUS
|
||
RsCompleteReads(IN PRP_FILE_CONTEXT Context);
|
||
|
||
|
||
NTSTATUS
|
||
RsPreserveDates(IN PRP_FILE_CONTEXT Context);
|
||
|
||
NTSTATUS
|
||
RsMarkUsn(IN PRP_FILE_CONTEXT Context);
|
||
|
||
NTSTATUS
|
||
RsOpenTarget(IN PRP_FILE_CONTEXT Context,
|
||
IN ULONG OpenAccess,
|
||
IN ULONG AdditionalAccess,
|
||
OUT HANDLE *Handle,
|
||
OUT PFILE_OBJECT *FileObject);
|
||
|
||
|
||
ULONG
|
||
RsIsNoRecall(IN PFILE_OBJECT fileObj,
|
||
OUT PRP_DATA *rpData);
|
||
|
||
NTSTATUS
|
||
RsPartialData(IN PDEVICE_OBJECT DeviceObject,
|
||
IN ULONGLONG filterId,
|
||
IN NTSTATUS status,
|
||
IN CHAR *buffer,
|
||
IN ULONG bytesRead,
|
||
IN ULONGLONG offset);
|
||
|
||
NTSTATUS
|
||
RsPartialWrite(IN PDEVICE_OBJECT DeviceObject,
|
||
IN PRP_FILE_CONTEXT Context,
|
||
IN CHAR *Buffer,
|
||
IN ULONG BufLen,
|
||
IN ULONGLONG Offset);
|
||
|
||
NTSTATUS
|
||
RsDoWrite(IN PDEVICE_OBJECT DeviceObject,
|
||
IN PRP_FILE_CONTEXT Context);
|
||
|
||
|
||
NTSTATUS
|
||
RsQueueRecall(IN ULONGLONG filterId,
|
||
IN ULONGLONG recallStart,
|
||
IN ULONGLONG recallSize);
|
||
|
||
|
||
NTSTATUS
|
||
RsQueueNoRecall(IN PFILE_OBJECT FileObject,
|
||
IN PIRP Irp,
|
||
IN ULONGLONG RecallStart,
|
||
IN ULONGLONG RecallSize,
|
||
IN ULONG BufferOffset,
|
||
IN ULONG BufferLength,
|
||
IN PRP_FILE_BUF CacheBuffer,
|
||
IN PVOID UserBuffer);
|
||
|
||
NTSTATUS
|
||
RsQueueNoRecallOpen(IN PRP_FILE_OBJ entry,
|
||
IN ULONGLONG filterId,
|
||
IN ULONGLONG offset,
|
||
IN ULONGLONG size);
|
||
NTSTATUS
|
||
RsQueueRecallOpen(IN PRP_FILE_CONTEXT Context,
|
||
IN PRP_FILE_OBJ Entry,
|
||
IN ULONGLONG FilterId,
|
||
IN ULONGLONG Offset,
|
||
IN ULONGLONG Size,
|
||
IN ULONG Command);
|
||
|
||
NTSTATUS
|
||
RsGetFileInfo(IN PRP_FILE_OBJ Entry,
|
||
IN PDEVICE_OBJECT DeviceObject);
|
||
|
||
NTSTATUS
|
||
RsGetFileId(IN PRP_FILE_OBJ entry,
|
||
IN PDEVICE_OBJECT DeviceObject);
|
||
|
||
NTSTATUS
|
||
RsGetFileName(IN PRP_FILE_OBJ entry,
|
||
IN PDEVICE_OBJECT DeviceObject);
|
||
|
||
NTSTATUS
|
||
RsCloseFile(IN ULONGLONG filterId);
|
||
|
||
NTSTATUS
|
||
RsCleanupFileObject(IN ULONGLONG filterId);
|
||
|
||
NTSTATUS
|
||
RsCompleteIrp(
|
||
IN PDEVICE_OBJECT DeviceObject,
|
||
IN PIRP Irp,
|
||
IN PVOID Context);
|
||
|
||
NTSTATUS
|
||
RsCheckRead(IN PIRP irp,
|
||
IN PFILE_OBJECT fileObject,
|
||
IN PDEVICE_EXTENSION deviceExtension);
|
||
|
||
NTSTATUS
|
||
RsCheckWrite(IN PIRP irp,
|
||
IN PFILE_OBJECT fileObject,
|
||
IN PDEVICE_EXTENSION deviceExtension);
|
||
|
||
NTSTATUS
|
||
RsFailAllRequests(IN PRP_FILE_CONTEXT Context,
|
||
IN BOOLEAN Norecall);
|
||
|
||
NTSTATUS
|
||
RsCompleteAllRequests(
|
||
IN PRP_FILE_CONTEXT Context,
|
||
IN PRP_FILE_OBJ Entry,
|
||
IN NTSTATUS Status
|
||
);
|
||
|
||
NTSTATUS
|
||
RsWriteReparsePointData(IN PRP_FILE_CONTEXT Context);
|
||
|
||
NTSTATUS
|
||
RsTruncateFile(IN PRP_FILE_CONTEXT Context);
|
||
|
||
|
||
NTSTATUS
|
||
RsSetEndOfFile(IN PRP_FILE_CONTEXT Context,
|
||
IN ULONGLONG size);
|
||
|
||
BOOLEAN
|
||
RsIsFastIoPossible(IN PFILE_OBJECT fileObj);
|
||
|
||
PIRP
|
||
RsGetFsaRequest(VOID);
|
||
|
||
PRP_FILE_OBJ
|
||
RsFindQueue(IN ULONGLONG filterId);
|
||
|
||
|
||
NTSTATUS
|
||
RsAddIo(IN PIRP irp);
|
||
|
||
PIRP
|
||
RsRemoveIo(VOID);
|
||
|
||
VOID
|
||
RsCompleteRead(IN PRP_IRP_QUEUE Irp,
|
||
IN BOOLEAN unlock);
|
||
|
||
|
||
BOOLEAN
|
||
RsIsFileObj(IN PFILE_OBJECT fileObj,
|
||
IN BOOLEAN returnContextData,
|
||
OUT PRP_DATA *rpData,
|
||
OUT POBJECT_NAME_INFORMATION *str,
|
||
OUT LONGLONG *fileId,
|
||
OUT LONGLONG *objIdHi,
|
||
OUT LONGLONG *objIdLo,
|
||
OUT ULONG *options,
|
||
OUT ULONGLONG *filterId,
|
||
OUT USN *usn);
|
||
|
||
VOID
|
||
RsCancelRecalls(VOID);
|
||
|
||
VOID
|
||
RsCancelIo(VOID);
|
||
|
||
VOID
|
||
RsLogValidateNeeded(IN ULONG serial);
|
||
|
||
BOOLEAN
|
||
RsAddValidateObj(IN ULONG serial,
|
||
IN LARGE_INTEGER cTime);
|
||
BOOLEAN
|
||
RsRemoveValidateObj(IN ULONG serial);
|
||
|
||
NTSTATUS
|
||
RsQueueValidate(IN ULONG serial);
|
||
|
||
ULONG
|
||
RsTerminate(VOID);
|
||
|
||
NTSTATUS
|
||
RsGetRecallInfo(IN OUT PRP_MSG Msg,
|
||
OUT PULONG_PTR InfoSize,
|
||
IN KPROCESSOR_MODE RequestorMode);
|
||
|
||
|
||
VOID
|
||
RsCancelReadRecall(IN PDEVICE_OBJECT DeviceObject,
|
||
IN PIRP Irp);
|
||
VOID
|
||
RsCancelWriteRecall(IN PDEVICE_OBJECT DeviceObject,
|
||
IN PIRP Irp);
|
||
|
||
VOID
|
||
RsLogError(IN ULONG line,
|
||
IN ULONG file,
|
||
IN ULONG code,
|
||
IN NTSTATUS ioError,
|
||
IN PIO_STACK_LOCATION irpSp,
|
||
IN WCHAR *msgString);
|
||
|
||
|
||
|
||
ULONG
|
||
RsGetReparseData(IN PFILE_OBJECT fileObject,
|
||
IN PDEVICE_OBJECT deviceObject,
|
||
OUT PRP_DATA rpData);
|
||
|
||
|
||
NTSTATUS
|
||
RsCheckVolumeReadOnly (IN PDEVICE_OBJECT FilterDeviceObject,
|
||
IN OUT PBOOLEAN pbReturnedFlagReadOnly);
|
||
|
||
|
||
NTSTATUS
|
||
RsQueryValueKey (
|
||
IN PUNICODE_STRING KeyName,
|
||
IN PUNICODE_STRING ValueName,
|
||
IN OUT PULONG ValueLength,
|
||
IN OUT PKEY_VALUE_FULL_INFORMATION *KeyValueInformation,
|
||
IN OUT PBOOLEAN DeallocateKeyValue);
|
||
|
||
|
||
NTSTATUS
|
||
RsCacheInitialize(
|
||
VOID
|
||
);
|
||
|
||
VOID
|
||
RsCacheFsaPartialData(
|
||
IN PRP_IRP_QUEUE ReadIo,
|
||
IN PUCHAR Buffer,
|
||
IN ULONGLONG Offset,
|
||
IN ULONG Length,
|
||
IN NTSTATUS Status
|
||
);
|
||
|
||
VOID
|
||
RsCacheFsaIoComplete(
|
||
IN PRP_IRP_QUEUE ReadIo,
|
||
IN NTSTATUS Status
|
||
);
|
||
|
||
NTSTATUS
|
||
RsGetNoRecallData(
|
||
IN PFILE_OBJECT FileObject,
|
||
IN PIRP Irp,
|
||
IN USN Usn,
|
||
IN LONGLONG FileOffset,
|
||
IN LONGLONG Length,
|
||
IN PUCHAR UserBuffer
|
||
);
|
||
|
||
LONG
|
||
RsExceptionFilter(
|
||
IN WCHAR *FunctionName,
|
||
IN PEXCEPTION_POINTERS ExceptionPointer);
|
||
|
||
NTSTATUS
|
||
RsTruncateOnClose(
|
||
IN PRP_FILE_CONTEXT Context
|
||
);
|
||
|
||
NTSTATUS
|
||
RsSetPremigratedState(IN PRP_FILE_CONTEXT Context);
|
||
|
||
NTSTATUS
|
||
RsDeleteReparsePoint(IN PRP_FILE_CONTEXT Context);
|
||
NTSTATUS
|
||
RsSetResetAttributes(IN PFILE_OBJECT FileObject,
|
||
IN ULONG SetAttributes,
|
||
IN ULONG ResetAttributes);
|
||
BOOLEAN
|
||
RsSetCancelRoutine(IN PIRP Irp,
|
||
IN PDRIVER_CANCEL CancelRoutine);
|
||
BOOLEAN
|
||
RsClearCancelRoutine (
|
||
IN PIRP Irp
|
||
);
|
||
NTSTATUS
|
||
RsGetFileUsn(IN PRP_FILE_CONTEXT Context,
|
||
IN PFILE_OBJECT FileObject,
|
||
IN PDEVICE_OBJECT FilterDeviceObject);
|
||
VOID
|
||
RsInterlockedRemoveEntryList(PLIST_ENTRY Entry,
|
||
PKSPIN_LOCK Lock);
|
||
VOID
|
||
RsGetUserInfo(
|
||
IN PSECURITY_SUBJECT_CONTEXT SubjectContext,
|
||
OUT PRP_USER_SECURITY_INFO UserSecurityInfo);
|
||
|
||
|
||
|
||
|
||
|
||
typedef enum _RpModuleCode
|
||
{
|
||
ModRpFilter = 100
|
||
,ModRpFilfun
|
||
,ModRpCache
|
||
,ModRpzw
|
||
,ModRpSec
|
||
} RpModuleCode;
|
||
|
||
typedef struct _RP_TRACE_ENTRY
|
||
{
|
||
RpModuleCode ModuleCode;
|
||
USHORT usLineNumber;
|
||
USHORT usIrql;
|
||
LARGE_INTEGER Timestamp;
|
||
ULONG_PTR Value1;
|
||
ULONG_PTR Value2;
|
||
ULONG_PTR Value3;
|
||
ULONG_PTR Value4;
|
||
} RP_TRACE_ENTRY, *PRP_TRACE_ENTRY;
|
||
|
||
|
||
typedef struct _RP_TRACE_CONTROL_BLOCK
|
||
{
|
||
KSPIN_LOCK Lock;
|
||
PRP_TRACE_ENTRY EntryBuffer;
|
||
ULONG EntryMaximum;
|
||
ULONG EntryNext;
|
||
} RP_TRACE_CONTROL_BLOCK, *PRP_TRACE_CONTROL_BLOCK;
|
||
|
||
|
||
|
||
#define RsTrace0(_ModuleCode) RsTrace4 ((_ModuleCode), 0, 0, 0, 0)
|
||
#define RsTrace1(_ModuleCode, _Value1) RsTrace4 ((_ModuleCode), (_Value1), 0, 0, 0)
|
||
#define RsTrace2(_ModuleCode, _Value1, _Value2) RsTrace4 ((_ModuleCode), (_Value1), (_Value2), 0, 0)
|
||
#define RsTrace3(_ModuleCode, _Value1, _Value2, _Value3) RsTrace4 ((_ModuleCode), (_Value1), (_Value2), (_Value3), 0)
|
||
|
||
#if DBG
|
||
#define RsTrace4(_ModuleCode, _Value1, _Value2, _Value3, _Value4) RsTraceAddEntry ((_ModuleCode), \
|
||
((USHORT)(__LINE__)), \
|
||
((ULONG_PTR)(_Value1)), \
|
||
((ULONG_PTR)(_Value2)), \
|
||
((ULONG_PTR)(_Value3)), \
|
||
((ULONG_PTR)(_Value4)))
|
||
#else
|
||
#define RsTrace4(_ModuleCode, _Value1, _Value2, _Value3, _Value4)
|
||
#endif
|
||
|
||
|
||
#if DBG
|
||
#define DEFAULT_TRACE_ENTRIES (0x4000)
|
||
#else
|
||
#define DEFAULT_TRACE_ENTRIES (0)
|
||
#endif
|
||
|
||
VOID RsTraceAddEntry (RpModuleCode ModuleCode,
|
||
USHORT usLineNumber,
|
||
ULONG_PTR Value1,
|
||
ULONG_PTR Value2,
|
||
ULONG_PTR Value3,
|
||
ULONG_PTR Value4);
|
||
|
||
NTSTATUS RsTraceInitialize (ULONG ulRequestedTraceEntries);
|
||
|
||
extern PRP_TRACE_CONTROL_BLOCK RsTraceControlBlock;
|
||
extern ULONG RsDefaultTraceEntries;
|
||
|
||
NTSTATUS RsLookupContext (PFILE_OBJECT pFileObject,
|
||
PRP_FILE_OBJ *pReturnedRpFileObject,
|
||
PRP_FILE_CONTEXT *pReturnedRpFileContext);
|