diff --git a/examples/memfs/src/main.rs b/examples/memfs/src/main.rs index a679259..730145a 100644 --- a/examples/memfs/src/main.rs +++ b/examples/memfs/src/main.rs @@ -6,11 +6,11 @@ use std::{ }; use winfsp_wrs::{ filetime_now, u16cstr, u16str, CleanupFlags, CreateFileInfo, CreateOptions, DirInfo, - FileAccessRights, FileAttributes, FileInfo, FileSystem, FileSystemContext, PSecurityDescriptor, - Params, SecurityDescriptor, U16CStr, U16CString, U16Str, VolumeInfo, VolumeParams, WriteMode, - NTSTATUS, STATUS_ACCESS_DENIED, STATUS_DIRECTORY_NOT_EMPTY, STATUS_END_OF_FILE, - STATUS_MEDIA_WRITE_PROTECTED, STATUS_NOT_A_DIRECTORY, STATUS_OBJECT_NAME_COLLISION, - STATUS_OBJECT_NAME_NOT_FOUND, + FileAccessRights, FileAttributes, FileInfo, FileSystem, FileSystemInterface, + PSecurityDescriptor, Params, SecurityDescriptor, U16CStr, U16CString, U16Str, VolumeInfo, + VolumeParams, WriteMode, NTSTATUS, STATUS_ACCESS_DENIED, STATUS_DIRECTORY_NOT_EMPTY, + STATUS_END_OF_FILE, STATUS_MEDIA_WRITE_PROTECTED, STATUS_NOT_A_DIRECTORY, + STATUS_OBJECT_NAME_COLLISION, STATUS_OBJECT_NAME_NOT_FOUND, }; macro_rules! debug { @@ -246,17 +246,19 @@ impl MemFs { } } -impl FileSystemContext for MemFs { +impl FileSystemInterface for MemFs { type FileContext = Arc>; const SET_DELETE_DEFINED: bool = true; + const GET_VOLUME_INFO_DEFINED: bool = true; fn get_volume_info(&self) -> Result { debug!("get_volume_info()"); Ok(self.volume_info.lock().unwrap().clone()) } + const SET_VOLUME_LABEL_DEFINED: bool = true; fn set_volume_label(&self, volume_label: &U16CStr) -> Result { debug!("set_volume_label(volume_label: {:?})", volume_label); @@ -269,6 +271,7 @@ impl FileSystemContext for MemFs { Ok(guard.clone()) } + const GET_SECURITY_BY_NAME_DEFINED: bool = true; fn get_security_by_name( &self, file_name: &U16CStr, @@ -298,6 +301,7 @@ impl FileSystemContext for MemFs { } } + const CREATE_EX_DEFINED: bool = true; fn create_ex( &self, file_name: &U16CStr, @@ -349,6 +353,7 @@ impl FileSystemContext for MemFs { Ok((file_context, file_info)) } + const OPEN_DEFINED: bool = true; fn open( &self, file_name: &U16CStr, @@ -372,6 +377,7 @@ impl FileSystemContext for MemFs { } } + const OVERWRITE_EX_DEFINED: bool = true; fn overwrite_ex( &self, file_context: Self::FileContext, @@ -416,6 +422,7 @@ impl FileSystemContext for MemFs { self.get_file_info_from_obj(&fc) } + const CLEANUP_DEFINED: bool = true; fn cleanup( &self, file_context: Self::FileContext, @@ -479,6 +486,7 @@ impl FileSystemContext for MemFs { } } + const READ_DEFINED: bool = true; fn read( &self, file_context: Self::FileContext, @@ -505,6 +513,7 @@ impl FileSystemContext for MemFs { } } + const WRITE_DEFINED: bool = true; fn write( &self, file_context: Self::FileContext, @@ -539,6 +548,7 @@ impl FileSystemContext for MemFs { Ok((written, self.get_file_info_from_obj(&fc)?)) } + const FLUSH_DEFINED: bool = true; fn flush(&self, file_context: Self::FileContext) -> Result { let fc = file_context.lock().unwrap(); debug!("[WinFSP] flush(file_context: {:?})", fc); @@ -546,6 +556,7 @@ impl FileSystemContext for MemFs { self.get_file_info_from_obj(&fc) } + const GET_FILE_INFO_DEFINED: bool = true; fn get_file_info(&self, file_context: Self::FileContext) -> Result { let fc = file_context.lock().unwrap(); debug!("[WinFSP] get_file_info(file_context: {:?})", fc); @@ -556,6 +567,7 @@ impl FileSystemContext for MemFs { } } + const SET_BASIC_INFO_DEFINED: bool = true; fn set_basic_info( &self, file_context: Self::FileContext, @@ -615,6 +627,7 @@ impl FileSystemContext for MemFs { self.get_file_info_from_obj(&fc) } + const SET_FILE_SIZE_DEFINED: bool = true; fn set_file_size( &self, file_context: Self::FileContext, @@ -647,6 +660,7 @@ impl FileSystemContext for MemFs { self.get_file_info_from_obj(&fc) } + const RENAME_DEFINED: bool = true; fn rename( &self, file_context: Self::FileContext, @@ -699,6 +713,7 @@ impl FileSystemContext for MemFs { Ok(()) } + const GET_SECURITY_DEFINED: bool = true; fn get_security( &self, file_context: Self::FileContext, @@ -712,6 +727,7 @@ impl FileSystemContext for MemFs { } } + const SET_SECURITY_DEFINED: bool = true; fn set_security( &self, file_context: Self::FileContext, @@ -743,6 +759,7 @@ impl FileSystemContext for MemFs { Ok(()) } + const READ_DIRECTORY_DEFINED: bool = true; fn read_directory( &self, file_context: Self::FileContext, @@ -808,6 +825,7 @@ impl FileSystemContext for MemFs { } } + const SET_DELETE_DEFINED: bool = true; fn set_delete( &self, file_context: Self::FileContext, diff --git a/examples/minimal/src/main.rs b/examples/minimal/src/main.rs index 44215ab..0ecb897 100644 --- a/examples/minimal/src/main.rs +++ b/examples/minimal/src/main.rs @@ -2,7 +2,7 @@ use std::sync::Arc; use winfsp_wrs::{ filetime_now, u16cstr, u16str, CreateOptions, DirInfo, FileAccessRights, FileAttributes, - FileInfo, FileSystem, FileSystemContext, PSecurityDescriptor, Params, SecurityDescriptor, + FileInfo, FileSystem, FileSystemInterface, PSecurityDescriptor, Params, SecurityDescriptor, U16CStr, U16Str, VolumeInfo, VolumeParams, NTSTATUS, }; @@ -48,9 +48,10 @@ impl MemFs { } } -impl FileSystemContext for MemFs { +impl FileSystemInterface for MemFs { type FileContext = Arc; + const GET_SECURITY_BY_NAME_DEFINED: bool = true; fn get_security_by_name( &self, _file_name: &U16CStr, @@ -63,6 +64,7 @@ impl FileSystemContext for MemFs { )) } + const OPEN_DEFINED: bool = true; fn open( &self, _file_name: &U16CStr, @@ -74,14 +76,17 @@ impl FileSystemContext for MemFs { Ok((file_context, file_info)) } + const GET_FILE_INFO_DEFINED: bool = true; fn get_file_info(&self, _file_context: Self::FileContext) -> Result { Ok(self.file_context.info) } + const GET_VOLUME_INFO_DEFINED: bool = true; fn get_volume_info(&self) -> Result { Ok(self.volume_info.clone()) } + const READ_DIRECTORY_DEFINED: bool = true; fn read_directory( &self, _file_context: Self::FileContext, diff --git a/winfsp_wrs/src/callback.rs b/winfsp_wrs/src/callback.rs index e027085..d0e65fa 100644 --- a/winfsp_wrs/src/callback.rs +++ b/winfsp_wrs/src/callback.rs @@ -1,7 +1,7 @@ use std::sync::Arc; use widestring::U16CStr; use windows_sys::Win32::Foundation::{ - STATUS_BUFFER_OVERFLOW, STATUS_NOT_IMPLEMENTED, STATUS_REPARSE, STATUS_SUCCESS, + STATUS_BUFFER_OVERFLOW, STATUS_REPARSE, STATUS_SUCCESS, }; use winfsp_wrs_sys::{ FspFileSystemAddDirInfo, FspFileSystemFindReparsePoint, FspFileSystemResolveReparsePoints, @@ -76,19 +76,91 @@ impl FileContextKind for usize { } } -pub trait FileSystemContext { +/// High level interface over `FSP_FILE_SYSTEM_INTERFACE`. +/// +/// This trait requires to overwrite all WinFSP callbacks you need and it corresponding +/// `xxx_DEFINED` associated const boolean. +/// +/// This is needed to properly build the `FSP_FILE_SYSTEM_INTERFACE` struct, as +/// a callback pointer set to `NULL` (i.e. if `xxx_DEFINED=false`) leads to a different +/// behavior that a callback pointer containing a mock implementation (e.g. +/// returning `Err(STATUS_NOT_IMPLEMENTED)`). +/// +/// So the way to work with this trait is to overwrite the method and `xxx_DEFINED` for +/// each function pointer you will need in `FSP_FILE_SYSTEM_INTERFACE`: +/// +/// ```rust +/// struct MyFS; +/// impl FileSystemInterface for MyFS { +/// type FileContext: usize; +/// // `CREATE_DEFINED` not overwritten, hence `FSP_FILE_SYSTEM_INTERFACE.Create == NULL` +/// const CREATE_EX_DEFINED: bool = true; // i.e. `FSP_FILE_SYSTEM_INTERFACE.CreateEx != NULL` +/// fn create_ex( +/// &self, +/// file_name: &U16CStr, +/// create_file_info: CreateFileInfo, +/// security_descriptor: SecurityDescriptor, +/// buffer: &[u8], +/// extra_buffer_is_reparse_point: bool, +/// ) -> Result<(Self::FileContext, FileInfo), NTSTATUS> { +/// ... +/// } +/// } +/// ``` +/// +/// *Notes*: +/// - Associated method and `xxx_DEFINED` const must be overwritten together, as the +/// method is simply ignored if `xxx_DEFINED` is not set, and setting `xxx_DEFINED` +/// without overwritting the method means the function pointer relies on the method +/// default implementation that panics whenever used (ah !). +/// - If your are curious about the reason for using a trait here instead of a struct (or +/// associated const fields with `Option` type in the trait instead of methods), it +/// all boils down to the fact some methods have an `impl Fn` function pointer as argument, +/// which is only possible in trait method. +pub trait FileSystemInterface { type FileContext: FileContextKind; - /// `SetDelete` takes precedence over `CanDelete` if both are defined in `FSP_FILE_SYSTEM_INTERFACE`. - /// If this boolean is not set, `FSP_FILE_SYSTEM_INTERFACE`'s `SetDelete` function will not be set. + const GET_VOLUME_INFO_DEFINED: bool = false; + const SET_VOLUME_LABEL_DEFINED: bool = false; + const GET_SECURITY_BY_NAME_DEFINED: bool = false; + const CREATE_DEFINED: bool = false; + const CREATE_EX_DEFINED: bool = false; + const OPEN_DEFINED: bool = false; + const OVERWRITE_DEFINED: bool = false; + const OVERWRITE_EX_DEFINED: bool = false; + const CLEANUP_DEFINED: bool = false; + const CLOSE_DEFINED: bool = false; + const READ_DEFINED: bool = false; + const WRITE_DEFINED: bool = false; + const FLUSH_DEFINED: bool = false; + const GET_FILE_INFO_DEFINED: bool = false; + const SET_BASIC_INFO_DEFINED: bool = false; + const SET_FILE_SIZE_DEFINED: bool = false; + const CAN_DELETE_DEFINED: bool = false; + const RENAME_DEFINED: bool = false; + const GET_SECURITY_DEFINED: bool = false; + const SET_SECURITY_DEFINED: bool = false; + const READ_DIRECTORY_DEFINED: bool = false; + const GET_REPARSE_POINT_DEFINED: bool = false; + const SET_REPARSE_POINT_DEFINED: bool = false; + const DELETE_REPARSE_POINT_DEFINED: bool = false; + const GET_STREAM_INFO_DEFINED: bool = false; + const GET_DIR_INFO_BY_NAME_DEFINED: bool = false; + const CONTROL_DEFINED: bool = false; const SET_DELETE_DEFINED: bool = false; + const GET_EA_DEFINED: bool = false; + const SET_EA_DEFINED: bool = false; + const DISPATCHER_STOPPED_DEFINED: bool = false; + const RESOLVE_REPARSE_POINTS_DEFINED: bool = false; /// Get volume information. - fn get_volume_info(&self) -> Result; + fn get_volume_info(&self) -> Result { + unreachable!("To be used, trait method must be overwritten !"); + } /// Set volume label. fn set_volume_label(&self, _volume_label: &U16CStr) -> Result { - Err(STATUS_NOT_IMPLEMENTED) + unreachable!("To be used, trait method must be overwritten !"); } /// Get file or directory attributes and security descriptor given a file name. @@ -105,11 +177,30 @@ pub trait FileSystemContext { /// set to `true`. fn get_security_by_name( &self, - file_name: &U16CStr, - find_reparse_point: impl Fn() -> Option, - ) -> Result<(FileAttributes, PSecurityDescriptor, bool), NTSTATUS>; + _file_name: &U16CStr, + _find_reparse_point: impl Fn() -> Option, + ) -> Result<(FileAttributes, PSecurityDescriptor, bool), NTSTATUS> { + unreachable!("To be used, trait method must be overwritten !"); + } + + /// Create new file or directory. + /// + /// Note: `FileSystemContext::create_ex` takes precedence over `FileSystemContext::create` + fn create( + &self, + _file_name: &U16CStr, + _create_file_info: CreateFileInfo, + _security_descriptor: SecurityDescriptor, + ) -> Result<(Self::FileContext, FileInfo), NTSTATUS> { + unreachable!("To be used, trait method must be overwritten !"); + } /// Create new file or directory. + /// + /// This function works like `create`, except that it also accepts an extra buffer + /// that may contain extended attributes or a reparse point. + /// + /// Note: `FileSystemContext::create_ex` takes precedence over `FileSystemContext::create` fn create_ex( &self, _file_name: &U16CStr, @@ -118,18 +209,37 @@ pub trait FileSystemContext { _buffer: &[u8], _extra_buffer_is_reparse_point: bool, ) -> Result<(Self::FileContext, FileInfo), NTSTATUS> { - Err(STATUS_NOT_IMPLEMENTED) + unreachable!("To be used, trait method must be overwritten !"); } /// Open a file or directory. fn open( &self, - file_name: &U16CStr, - create_options: CreateOptions, - granted_access: FileAccessRights, - ) -> Result<(Self::FileContext, FileInfo), NTSTATUS>; + _file_name: &U16CStr, + _create_options: CreateOptions, + _granted_access: FileAccessRights, + ) -> Result<(Self::FileContext, FileInfo), NTSTATUS> { + unreachable!("To be used, trait method must be overwritten !"); + } + + /// Overwrite a file. + /// + /// Note: `FileSystemContext::overwrite_ex` takes precedence over `FileSystemContext::overwrite` + fn overwrite( + &self, + _file_context: Self::FileContext, + _file_attributes: FileAttributes, + _replace_file_attributes: bool, + _allocation_size: u64, + ) -> Result { + unreachable!("To be used, trait method must be overwritten !"); + } /// Overwrite a file. + /// + /// This function works like `overwrite`, except that it also accepts EA (extended attributes). + /// + /// Note: `FileSystemContext::overwrite_ex` takes precedence over `FileSystemContext::overwrite` fn overwrite_ex( &self, _file_context: Self::FileContext, @@ -138,7 +248,7 @@ pub trait FileSystemContext { _allocation_size: u64, _buffer: &[u8], ) -> Result { - Err(STATUS_NOT_IMPLEMENTED) + unreachable!("To be used, trait method must be overwritten !"); } /// Cleanup a file. @@ -148,10 +258,13 @@ pub trait FileSystemContext { _file_name: Option<&U16CStr>, _flags: CleanupFlags, ) { + unreachable!("To be used, trait method must be overwritten !"); } /// Close a file. - fn close(&self, _file_context: Self::FileContext) {} + fn close(&self, _file_context: Self::FileContext) { + unreachable!("To be used, trait method must be overwritten !"); + } /// Read a file. fn read( @@ -160,7 +273,7 @@ pub trait FileSystemContext { _buffer: &mut [u8], _offset: u64, ) -> Result { - Err(STATUS_NOT_IMPLEMENTED) + unreachable!("To be used, trait method must be overwritten !"); } /// Write a file. @@ -170,16 +283,18 @@ pub trait FileSystemContext { _buffer: &[u8], _mode: WriteMode, ) -> Result<(usize, FileInfo), NTSTATUS> { - Err(STATUS_NOT_IMPLEMENTED) + unreachable!("To be used, trait method must be overwritten !"); } /// Flush a file or volume. fn flush(&self, _file_context: Self::FileContext) -> Result { - Err(STATUS_NOT_IMPLEMENTED) + unreachable!("To be used, trait method must be overwritten !"); } /// Get file or directory information. - fn get_file_info(&self, file_context: Self::FileContext) -> Result; + fn get_file_info(&self, _file_context: Self::FileContext) -> Result { + unreachable!("To be used, trait method must be overwritten !"); + } /// Set file or directory basic information. fn set_basic_info( @@ -191,7 +306,7 @@ pub trait FileSystemContext { _last_write_time: u64, _change_time: u64, ) -> Result { - Err(STATUS_NOT_IMPLEMENTED) + unreachable!("To be used, trait method must be overwritten !"); } /// Set file/allocation size. @@ -201,16 +316,18 @@ pub trait FileSystemContext { _new_size: u64, _set_allocation_size: bool, ) -> Result { - Err(STATUS_NOT_IMPLEMENTED) + unreachable!("To be used, trait method must be overwritten !"); } /// Determine whether a file or directory can be deleted. + /// + /// Note: `FileSystemContext::set_delete` takes precedence over `FileSystemContext::can_delete` fn can_delete( &self, _file_context: Self::FileContext, _file_name: &U16CStr, ) -> Result<(), NTSTATUS> { - Err(STATUS_NOT_IMPLEMENTED) + unreachable!("To be used, trait method must be overwritten !"); } /// Renames a file or directory. @@ -221,7 +338,7 @@ pub trait FileSystemContext { _new_file_name: &U16CStr, _replace_if_exists: bool, ) -> Result<(), NTSTATUS> { - Err(STATUS_NOT_IMPLEMENTED) + unreachable!("To be used, trait method must be overwritten !"); } /// Get file or directory security descriptor. @@ -229,7 +346,7 @@ pub trait FileSystemContext { &self, _file_context: Self::FileContext, ) -> Result { - Err(STATUS_NOT_IMPLEMENTED) + unreachable!("To be used, trait method must be overwritten !"); } /// Set file or directory security descriptor. @@ -239,7 +356,7 @@ pub trait FileSystemContext { _security_information: u32, _modification_descriptor: PSecurityDescriptor, ) -> Result<(), NTSTATUS> { - Err(STATUS_NOT_IMPLEMENTED) + unreachable!("To be used, trait method must be overwritten !"); } /// Read a directory. @@ -247,10 +364,12 @@ pub trait FileSystemContext { /// `add_dir_info` returns `false` if there is no more space left to add elements. fn read_directory( &self, - file_context: Self::FileContext, - marker: Option<&U16CStr>, - add_dir_info: impl FnMut(DirInfo) -> bool, - ) -> Result<(), NTSTATUS>; + _file_context: Self::FileContext, + _marker: Option<&U16CStr>, + _add_dir_info: impl FnMut(DirInfo) -> bool, + ) -> Result<(), NTSTATUS> { + unreachable!("To be used, trait method must be overwritten !"); + } /// Get reparse point. fn get_reparse_point( @@ -259,7 +378,7 @@ pub trait FileSystemContext { _file_name: &U16CStr, _buffer: &mut [u8], ) -> Result { - Err(STATUS_NOT_IMPLEMENTED) + unreachable!("To be used, trait method must be overwritten !"); } /// Set reparse point. @@ -269,7 +388,7 @@ pub trait FileSystemContext { _file_name: &U16CStr, _buffer: &mut [u8], ) -> Result<(), NTSTATUS> { - Err(STATUS_NOT_IMPLEMENTED) + unreachable!("To be used, trait method must be overwritten !"); } /// Delete reparse point. @@ -279,7 +398,7 @@ pub trait FileSystemContext { _file_name: &U16CStr, _buffer: &mut [u8], ) -> Result<(), NTSTATUS> { - Err(STATUS_NOT_IMPLEMENTED) + unreachable!("To be used, trait method must be overwritten !"); } /// Get named streams information. @@ -288,7 +407,7 @@ pub trait FileSystemContext { _file_context: Self::FileContext, _buffer: &mut [u8], ) -> Result { - Err(STATUS_NOT_IMPLEMENTED) + unreachable!("To be used, trait method must be overwritten !"); } /// Get directory information for a single file or directory within a parent @@ -298,7 +417,7 @@ pub trait FileSystemContext { _file_context: Self::FileContext, _file_name: &U16CStr, ) -> Result { - Err(STATUS_NOT_IMPLEMENTED) + unreachable!("To be used, trait method must be overwritten !"); } /// Process control code. @@ -309,22 +428,24 @@ pub trait FileSystemContext { _input_buffer: &[u8], _output_buffer: &mut [u8], ) -> Result { - Err(STATUS_NOT_IMPLEMENTED) + unreachable!("To be used, trait method must be overwritten !"); } /// Set the file delete flag. + /// + /// Note: `FileSystemContext::set_delete` takes precedence over `FileSystemContext::can_delete` fn set_delete( &self, _file_context: Self::FileContext, _file_name: &U16CStr, _delete_file: bool, ) -> Result<(), NTSTATUS> { - Err(STATUS_NOT_IMPLEMENTED) + unreachable!("To be used, trait method must be overwritten !"); } /// Get extended attributes. fn get_ea(&self, _file_context: Self::FileContext, _buffer: &[u8]) -> Result { - Err(STATUS_NOT_IMPLEMENTED) + unreachable!("To be used, trait method must be overwritten !"); } /// Set extended attributes. @@ -333,30 +454,38 @@ pub trait FileSystemContext { _file_context: Self::FileContext, _buffer: &[u8], ) -> Result { - Err(STATUS_NOT_IMPLEMENTED) + unreachable!("To be used, trait method must be overwritten !"); + } + + fn dispatcher_stopped(&self, _normally: bool) { + unreachable!("To be used, trait method must be overwritten !"); } /// Get reparse point given a file name. + /// + /// This method is used as a callback parameter to `FspFileSystemFindReparsePoint` & + /// `FspFileSystemResolveReparsePoints` helpers to respectively implement + /// `FSP_FILE_SYSTEM_INTERFACE`'s `GetSecurityByName` & `ResolveReparsePoints`. fn get_reparse_point_by_name( &self, _file_name: &U16CStr, _is_directory: bool, _buffer: Option<&mut [u8]>, ) -> Result { - Err(STATUS_NOT_IMPLEMENTED) + unreachable!("To be used, trait method must be overwritten !"); } - - fn dispatcher_stopped(&self, _normally: bool) {} } -pub(crate) struct Interface; +/// `TrampolineInterface` fills the gap between the high level `FileSystemInterface` +/// and the `FSP_FILE_SYSTEM_INTERFACE` C struct that WinFSP expects from us. +pub(crate) struct TrampolineInterface; -impl Interface { +impl TrampolineInterface { /// Get volume information. /// - FileSystem - The file system on which this request is posted. /// - VolumeInfo - [out] Pointer to a structure that will receive the volume /// information on successful return from this call. - unsafe extern "C" fn get_volume_info_ext( + unsafe extern "C" fn get_volume_info_ext( file_system: *mut FSP_FILE_SYSTEM, volume_info: *mut FSP_FSCTL_VOLUME_INFO, ) -> NTSTATUS { @@ -376,7 +505,7 @@ impl Interface { /// - VolumeLabel - The new label for the volume. /// - VolumeInfo - [out] Pointer to a structure that will receive the volume /// information on successful return from this call. - unsafe extern "C" fn set_volume_label_w_ext( + unsafe extern "C" fn set_volume_label_w_ext( file_system: *mut FSP_FILE_SYSTEM, volume_label: PWSTR, volume_info: *mut FSP_FSCTL_VOLUME_INFO, @@ -413,7 +542,7 @@ impl Interface { /// Remarks: STATUS_REPARSE should be returned by file systems that support /// reparse points when they encounter a FileName that contains reparse points /// anywhere but the final path component. - unsafe extern "C" fn get_security_by_name_ext( + unsafe extern "C" fn get_security_by_name_ext( file_system: *mut FSP_FILE_SYSTEM, file_name: PWSTR, p_file_attributes: PUINT32, @@ -494,7 +623,7 @@ impl Interface { /// - FileInfo - [out] Pointer to a structure that will receive the file /// information on successful return from this call. This information /// includes file attributes, file times, etc. - unsafe extern "C" fn open_ext( + unsafe extern "C" fn open_ext( file_system: *mut FSP_FILE_SYSTEM, file_name: PWSTR, create_options: UINT32, @@ -527,7 +656,7 @@ impl Interface { /// Delete is requested. /// - Flags - These flags determine whether the file was modified and whether /// to delete the file. - unsafe extern "C" fn cleanup_ext( + unsafe extern "C" fn cleanup_ext( file_system: *mut FSP_FILE_SYSTEM, file_context: PVOID, file_name: PWSTR, @@ -548,7 +677,7 @@ impl Interface { /// Close a file. /// - FileSystem - The file system on which this request is posted. /// - FileContext - The file context of the file or directory to be closed. - unsafe extern "C" fn close_ext( + unsafe extern "C" fn close_ext( file_system: *mut FSP_FILE_SYSTEM, file_context: PVOID, ) { @@ -566,7 +695,7 @@ impl Interface { /// - Length - Length of data to read. /// - PBytesTransferred - [out] Pointer to a memory location that will receive /// the actual number of bytes read. - unsafe extern "C" fn read_ext( + unsafe extern "C" fn read_ext( file_system: *mut FSP_FILE_SYSTEM, file_context: PVOID, buffer: PVOID, @@ -602,7 +731,7 @@ impl Interface { /// - FileInfo - [out] Pointer to a structure that will receive the file /// information on successful return from this call. This information /// includes file attributes, file times, etc. - unsafe extern "C" fn write_ext( + unsafe extern "C" fn write_ext( file_system: *mut FSP_FILE_SYSTEM, file_context: PVOID, buffer: PVOID, @@ -645,7 +774,7 @@ impl Interface { /// information on successful return from this call. This information /// includes file attributes, file times, etc. Used when flushing file (not /// volume). - unsafe extern "C" fn flush_ext( + unsafe extern "C" fn flush_ext( file_system: *mut FSP_FILE_SYSTEM, file_context: PVOID, file_info: *mut FSP_FSCTL_FILE_INFO, @@ -669,7 +798,7 @@ impl Interface { /// - FileInfo - [out] Pointer to a structure that will receive the file /// information on successful return from this call. This information /// includes file attributes, file times, etc. - unsafe extern "C" fn get_file_info_ext( + unsafe extern "C" fn get_file_info_ext( file_system: *mut FSP_FILE_SYSTEM, file_context: PVOID, file_info: *mut FSP_FSCTL_FILE_INFO, @@ -704,7 +833,7 @@ impl Interface { /// - FileInfo - [out] Pointer to a structure that will receive the file /// information on successful return from this call. This information /// includes file attributes, file times, etc. - unsafe extern "C" fn set_basic_info_ext( + unsafe extern "C" fn set_basic_info_ext( file_system: *mut FSP_FILE_SYSTEM, file_context: PVOID, file_attributes: UINT32, @@ -744,7 +873,7 @@ impl Interface { /// - FileInfo - [out] Pointer to a structure that will receive the file /// information on successful return from this call. This information /// includes file attributes, file times, etc. - unsafe extern "C" fn set_file_size_ext( + unsafe extern "C" fn set_file_size_ext( file_system: *mut FSP_FILE_SYSTEM, file_context: PVOID, new_size: UINT64, @@ -770,7 +899,7 @@ impl Interface { /// Delete is requested. /// - Flags - These flags determine whether the file was modified and whether /// to delete the file. - unsafe extern "C" fn can_delete_ext( + unsafe extern "C" fn can_delete_ext( file_system: *mut FSP_FILE_SYSTEM, file_context: PVOID, file_name: PWSTR, @@ -792,7 +921,7 @@ impl Interface { /// - NewFileName - The new name for the file or directory. /// - ReplaceIfExists - Whether to replace a file that already exists at /// NewFileName. - unsafe extern "C" fn rename_ext( + unsafe extern "C" fn rename_ext( file_system: *mut FSP_FILE_SYSTEM, file_context: PVOID, file_name: PWSTR, @@ -820,7 +949,7 @@ impl Interface { /// buffer size. On input it contains the size of the security descriptor /// buffer. On output it will contain the actual size of the security /// descriptor copied into the security descriptor buffer. Cannot be NULL. - unsafe extern "C" fn get_security_ext( + unsafe extern "C" fn get_security_ext( file_system: *mut FSP_FILE_SYSTEM, file_context: PVOID, security_descriptor: PSECURITY_DESCRIPTOR, @@ -855,7 +984,7 @@ impl Interface { /// security descriptor should be modified. /// - ModificationDescriptor - Describes the modifications to apply to the file /// or directory security descriptor. - unsafe extern "C" fn set_security_ext( + unsafe extern "C" fn set_security_ext( file_system: *mut FSP_FILE_SYSTEM, file_context: PVOID, security_information: SECURITY_INFORMATION, @@ -887,7 +1016,7 @@ impl Interface { /// - Length - Length of data to read. /// - PBytesTransferred - [out] Pointer to a memory location that will receive /// the actual number of bytes read. - unsafe extern "C" fn read_directory_ext( + unsafe extern "C" fn read_directory_ext( file_system: *mut FSP_FILE_SYSTEM, file_context: PVOID, _pattern: PWSTR, @@ -936,7 +1065,7 @@ impl Interface { } } - unsafe extern "C" fn get_reparse_point_by_name_ext( + unsafe extern "C" fn get_reparse_point_by_name_ext( file_system: *mut FSP_FILE_SYSTEM, _context: PVOID, file_name: PWSTR, @@ -982,7 +1111,7 @@ impl Interface { /// - PSize - [in,out] Pointer to the buffer size. On input it contains the /// size of the buffer. On output it will contain the actual size of data /// copied. - unsafe extern "C" fn resolve_reparse_points_ext( + unsafe extern "C" fn resolve_reparse_points_ext( file_system: *mut FSP_FILE_SYSTEM, file_name: PWSTR, reparse_point_index: UINT32, @@ -1014,7 +1143,7 @@ impl Interface { /// - PSize - [in,out] Pointer to the buffer size. On input it contains the /// size of the buffer. On output it will contain the actual size of data /// copied. - unsafe extern "C" fn get_reparse_point_ext( + unsafe extern "C" fn get_reparse_point_ext( file_system: *mut FSP_FILE_SYSTEM, file_context: PVOID, file_name: PWSTR, @@ -1043,7 +1172,7 @@ impl Interface { /// If this buffer contains a symbolic link path, it should not be assumed to /// be NULL terminated. /// - Size - Size of data to write. - unsafe extern "C" fn set_reparse_point_ext( + unsafe extern "C" fn set_reparse_point_ext( file_system: *mut FSP_FILE_SYSTEM, file_context: PVOID, file_name: PWSTR, @@ -1067,7 +1196,7 @@ impl Interface { /// - FileName - The file name of the reparse point. /// - Buffer - Pointer to a buffer that contains the data for this operation. /// - Size - Size of data to write. - unsafe extern "C" fn delete_reparse_point_ext( + unsafe extern "C" fn delete_reparse_point_ext( file_system: *mut FSP_FILE_SYSTEM, file_context: PVOID, file_name: PWSTR, @@ -1093,7 +1222,7 @@ impl Interface { /// - Length - Length of buffer. /// - PBytesTransferred - [out] Pointer to a memory location that will receive /// the actual number of bytes stored. - unsafe extern "C" fn get_stream_info_ext( + unsafe extern "C" fn get_stream_info_ext( file_system: *mut FSP_FILE_SYSTEM, file_context: PVOID, buffer: PVOID, @@ -1122,7 +1251,7 @@ impl Interface { /// - DirInfo - [out] Pointer to a structure that will receive the directory /// information on successful return from this call. This information /// includes the file name, but also file attributes, file times, etc. - unsafe extern "C" fn get_dir_info_by_name_ext( + unsafe extern "C" fn get_dir_info_by_name_ext( file_system: *mut FSP_FILE_SYSTEM, file_context: PVOID, file_name: PWSTR, @@ -1160,7 +1289,7 @@ impl Interface { /// - OutputBufferLength - Output data length. /// - PBytesTransferred - [out] Pointer to a memory location that will receive /// the actual number of bytes transferred. - unsafe extern "C" fn control_ext( + unsafe extern "C" fn control_ext( file_system: *mut FSP_FILE_SYSTEM, file_context: PVOID, control_code: UINT32, @@ -1194,7 +1323,7 @@ impl Interface { /// deleted on Cleanup; otherwise it will not be deleted. It is legal to /// receive multiple SetDelete calls for the same file with different /// DeleteFile parameters. - unsafe extern "C" fn set_delete_ext( + unsafe extern "C" fn set_delete_ext( file_system: *mut FSP_FILE_SYSTEM, file_context: PVOID, file_name: PWSTR, @@ -1210,6 +1339,69 @@ impl Interface { } } + /// Create new file or directory. + /// - FileSystem - The file system on which this request is posted. + /// - FileName - The name of the file or directory to be created. + /// - CreateOptions - Create options for this request. This parameter has the + /// same meaning as the CreateOptions parameter of the NtCreateFile API. User + /// mode file systems should typically only be concerned with the flag + /// FILE_DIRECTORY_FILE, which is an instruction to create a directory rather + /// than a file. Some file systems may also want to pay attention to the + /// FILE_NO_INTERMEDIATE_BUFFERING and FILE_WRITE_THROUGH flags, although + /// these are typically handled by the FSD component. + /// - GrantedAccess - Determines the specific access rights that have been + /// granted for this request. Upon receiving this call all access checks have + /// been performed and the user mode file system need not perform any + /// additional checks. However this parameter may be useful to a user mode + /// file system; for example the WinFsp-FUSE layer uses this parameter to + /// determine which flags to use in its POSIX open() call. + /// - FileAttributes - File attributes to apply to the newly created file or + /// directory. + /// - SecurityDescriptor - Security descriptor to apply to the newly created + /// file or directory. This security descriptor will always be in + /// self-relative format. Its length can be retrieved using the Windows + /// GetSecurityDescriptorLength API. Will be NULL for named streams. + /// - AllocationSize - Allocation size for the newly created file. + /// - PFileContext - [out] Pointer that will receive the file context on + /// successful return from this call. + /// - FileInfo - [out] Pointer to a structure that will receive the file + /// information on successful return from this call. This information + /// includes file attributes, file times, etc. + unsafe extern "C" fn create_ext( + file_system: *mut FSP_FILE_SYSTEM, + file_name: PWSTR, + create_options: UINT32, + granted_access: UINT32, + file_attributes: UINT32, + security_descriptor: PSECURITY_DESCRIPTOR, + allocation_size: UINT64, + p_file_context: *mut PVOID, + file_info: *mut FSP_FSCTL_FILE_INFO, + ) -> NTSTATUS { + let fs = &*(*file_system).UserContext.cast::(); + let file_name = U16CStr::from_ptr_str(file_name); + let sd = SecurityDescriptor::from_ptr(security_descriptor); + + match C::create( + fs, + file_name, + CreateFileInfo { + create_options: CreateOptions(create_options), + granted_access: FileAccessRights(granted_access), + file_attributes: FileAttributes(file_attributes), + allocation_size, + }, + sd, + ) { + Ok((fctx, finfo)) => { + C::FileContext::write(fctx, p_file_context); + *file_info = finfo.0; + STATUS_SUCCESS + } + Err(e) => e, + } + } + /// Create new file or directory. /// - FileSystem - The file system on which this request is posted. /// - FileName - The name of the file or directory to be created. @@ -1242,7 +1434,7 @@ impl Interface { /// - FileInfo - [out] Pointer to a structure that will receive the file /// information on successful return from this call. This information /// includes file attributes, file times, etc. - unsafe extern "C" fn create_ex_ext( + unsafe extern "C" fn create_ex_ext( file_system: *mut FSP_FILE_SYSTEM, file_name: PWSTR, create_options: UINT32, @@ -1283,6 +1475,43 @@ impl Interface { } } + /// Overwrite a file. + /// - FileSystem - The file system on which this request is posted. + /// - FileContext - The file context of the file to overwrite. + /// - FileAttributes - File attributes to apply to the overwritten file. + /// - ReplaceFileAttributes - When TRUE the existing file attributes should be + /// replaced with the new ones. When FALSE the existing file attributes + /// should be merged (or'ed) with the new ones. + /// - AllocationSize - Allocation size for the overwritten file. + /// - FileInfo - [out] Pointer to a structure that will receive the file + /// information on successful return from this call. This information + /// includes file attributes, file times, etc. + unsafe extern "C" fn overwrite_ext( + file_system: *mut FSP_FILE_SYSTEM, + file_context: PVOID, + file_attributes: UINT32, + replace_file_attributes: BOOLEAN, + allocation_size: UINT64, + file_info: *mut FSP_FSCTL_FILE_INFO, + ) -> NTSTATUS { + let fs = &*(*file_system).UserContext.cast::(); + let fctx = C::FileContext::access(file_context); + + match C::overwrite( + fs, + fctx, + FileAttributes(file_attributes), + replace_file_attributes != 0, + allocation_size, + ) { + Ok(finfo) => { + *file_info = finfo.0; + STATUS_SUCCESS + } + Err(e) => e, + } + } + /// Overwrite a file. /// - FileSystem - The file system on which this request is posted. /// - FileContext - The file context of the file to overwrite. @@ -1296,7 +1525,7 @@ impl Interface { /// - FileInfo - [out] Pointer to a structure that will receive the file /// information on successful return from this call. This information /// includes file attributes, file times, etc. - unsafe extern "C" fn overwrite_ex_ext( + unsafe extern "C" fn overwrite_ex_ext( file_system: *mut FSP_FILE_SYSTEM, file_context: PVOID, file_attributes: UINT32, @@ -1334,7 +1563,7 @@ impl Interface { /// - EaLength - Extended attributes buffer length. /// - PBytesTransferred - [out] Pointer to a memory location that will receive /// the actual number of bytes transferred. - unsafe extern "C" fn get_ea_ext( + unsafe extern "C" fn get_ea_ext( file_system: *mut FSP_FILE_SYSTEM, file_context: PVOID, ea: PFILE_FULL_EA_INFORMATION, @@ -1363,7 +1592,7 @@ impl Interface { /// - FileInfo - [out] Pointer to a structure that will receive the file /// information on successful return from this call. This information /// includes file attributes, file times, etc. - unsafe extern "C" fn set_ea_ext( + unsafe extern "C" fn set_ea_ext( file_system: *mut FSP_FILE_SYSTEM, file_context: PVOID, ea: PFILE_FULL_EA_INFORMATION, @@ -1383,7 +1612,7 @@ impl Interface { } } - unsafe extern "C" fn dispatcher_stopped_ext( + unsafe extern "C" fn dispatcher_stopped_ext( file_system: *mut FSP_FILE_SYSTEM, normally: BOOLEAN, ) { @@ -1394,43 +1623,76 @@ impl Interface { FspFileSystemStopServiceIfNecessary(file_system, normally) } - pub(crate) fn interface() -> FSP_FILE_SYSTEM_INTERFACE { - let mut fsp_interface = FSP_FILE_SYSTEM_INTERFACE { - CanDelete: Some(Self::can_delete_ext::), - Cleanup: Some(Self::cleanup_ext::), - Close: Some(Self::close_ext::), - Control: Some(Self::control_ext::), - CreateEx: Some(Self::create_ex_ext::), - DeleteReparsePoint: Some(Self::delete_reparse_point_ext::), - DispatcherStopped: Some(Self::dispatcher_stopped_ext::), - Flush: Some(Self::flush_ext::), - GetDirInfoByName: Some(Self::get_dir_info_by_name_ext::), - GetEa: Some(Self::get_ea_ext::), - GetFileInfo: Some(Self::get_file_info_ext::), - GetReparsePoint: Some(Self::get_reparse_point_ext::), - GetSecurity: Some(Self::get_security_ext::), - GetSecurityByName: Some(Self::get_security_by_name_ext::), - GetStreamInfo: Some(Self::get_stream_info_ext::), - GetVolumeInfo: Some(Self::get_volume_info_ext::), - Open: Some(Self::open_ext::), - OverwriteEx: Some(Self::overwrite_ex_ext::), - Read: Some(Self::read_ext::), - ReadDirectory: Some(Self::read_directory_ext::), - Rename: Some(Self::rename_ext::), - ResolveReparsePoints: Some(Self::resolve_reparse_points_ext::), - SetBasicInfo: Some(Self::set_basic_info_ext::), - SetDelete: None, - SetEa: Some(Self::set_ea_ext::), - SetFileSize: Some(Self::set_file_size_ext::), - SetReparsePoint: Some(Self::set_reparse_point_ext::), - SetSecurity: Some(Self::set_security_ext::), - SetVolumeLabelW: Some(Self::set_volume_label_w_ext::), - Write: Some(Self::write_ext::), - ..Default::default() - }; - if Ctx::SET_DELETE_DEFINED { - fsp_interface.SetDelete = Some(Self::set_delete_ext::); + pub(crate) fn interface() -> FSP_FILE_SYSTEM_INTERFACE { + macro_rules! set_fn_pointer_or_null { + ($flag_name:ident, $fn_ext_name:ident) => { + if Ctx::$flag_name { + Some(Self::$fn_ext_name::) + } else { + None + } + }; + } + + FSP_FILE_SYSTEM_INTERFACE { + GetVolumeInfo: set_fn_pointer_or_null!(GET_VOLUME_INFO_DEFINED, get_volume_info_ext), + SetVolumeLabelW: set_fn_pointer_or_null!( + SET_VOLUME_LABEL_DEFINED, + set_volume_label_w_ext + ), + GetSecurityByName: set_fn_pointer_or_null!( + GET_SECURITY_BY_NAME_DEFINED, + get_security_by_name_ext + ), + Create: set_fn_pointer_or_null!(CREATE_DEFINED, create_ext), + CreateEx: set_fn_pointer_or_null!(CREATE_EX_DEFINED, create_ex_ext), + Open: set_fn_pointer_or_null!(OPEN_DEFINED, open_ext), + Overwrite: set_fn_pointer_or_null!(OVERWRITE_DEFINED, overwrite_ext), + OverwriteEx: set_fn_pointer_or_null!(OVERWRITE_EX_DEFINED, overwrite_ex_ext), + Cleanup: set_fn_pointer_or_null!(CLEANUP_DEFINED, cleanup_ext), + Close: set_fn_pointer_or_null!(CLOSE_DEFINED, close_ext), + Read: set_fn_pointer_or_null!(READ_DEFINED, read_ext), + Write: set_fn_pointer_or_null!(WRITE_DEFINED, write_ext), + Flush: set_fn_pointer_or_null!(FLUSH_DEFINED, flush_ext), + GetFileInfo: set_fn_pointer_or_null!(GET_FILE_INFO_DEFINED, get_file_info_ext), + SetBasicInfo: set_fn_pointer_or_null!(SET_BASIC_INFO_DEFINED, set_basic_info_ext), + SetFileSize: set_fn_pointer_or_null!(SET_FILE_SIZE_DEFINED, set_file_size_ext), + CanDelete: set_fn_pointer_or_null!(CAN_DELETE_DEFINED, can_delete_ext), + Rename: set_fn_pointer_or_null!(RENAME_DEFINED, rename_ext), + GetSecurity: set_fn_pointer_or_null!(GET_SECURITY_DEFINED, get_security_ext), + SetSecurity: set_fn_pointer_or_null!(SET_SECURITY_DEFINED, set_security_ext), + ReadDirectory: set_fn_pointer_or_null!(READ_DIRECTORY_DEFINED, read_directory_ext), + GetReparsePoint: set_fn_pointer_or_null!( + GET_REPARSE_POINT_DEFINED, + get_reparse_point_ext + ), + SetReparsePoint: set_fn_pointer_or_null!( + SET_REPARSE_POINT_DEFINED, + set_reparse_point_ext + ), + DeleteReparsePoint: set_fn_pointer_or_null!( + DELETE_REPARSE_POINT_DEFINED, + delete_reparse_point_ext + ), + GetStreamInfo: set_fn_pointer_or_null!(GET_STREAM_INFO_DEFINED, get_stream_info_ext), + GetDirInfoByName: set_fn_pointer_or_null!( + GET_DIR_INFO_BY_NAME_DEFINED, + get_dir_info_by_name_ext + ), + Control: set_fn_pointer_or_null!(CONTROL_DEFINED, control_ext), + SetDelete: set_fn_pointer_or_null!(SET_DELETE_DEFINED, set_delete_ext), + GetEa: set_fn_pointer_or_null!(GET_EA_DEFINED, get_ea_ext), + SetEa: set_fn_pointer_or_null!(SET_EA_DEFINED, set_ea_ext), + DispatcherStopped: set_fn_pointer_or_null!( + DISPATCHER_STOPPED_DEFINED, + dispatcher_stopped_ext + ), + ResolveReparsePoints: set_fn_pointer_or_null!( + RESOLVE_REPARSE_POINTS_DEFINED, + resolve_reparse_points_ext + ), + + ..Default::default() // Initializing `Obsolete0` & `Reserved` fields } - fsp_interface } } diff --git a/winfsp_wrs/src/file_system.rs b/winfsp_wrs/src/file_system.rs index 3e270fc..4b8aac8 100644 --- a/winfsp_wrs/src/file_system.rs +++ b/winfsp_wrs/src/file_system.rs @@ -20,7 +20,7 @@ use winfsp_wrs_sys::{ FSP_FSCTL_VOLUME_PARAMS, NTSTATUS, }; -use crate::{FileContextKind, FileSystemContext, Interface}; +use crate::{FileContextKind, FileSystemInterface, TrampolineInterface}; #[cfg(feature = "icon")] use crate::{FileAccessRights, FileAttributes, FileCreationDisposition, FileShareMode}; @@ -317,7 +317,7 @@ pub struct Params { } #[derive(Debug, Clone)] -pub struct FileSystem { +pub struct FileSystem { // FileSystem inner value inner: FSP_FILE_SYSTEM, pub params: Params, @@ -327,9 +327,9 @@ pub struct FileSystem { // SAFETY: FSP_FILE_SYSTEM contains `*mut c_void` pointers that cannot be send between threads // by default. However this structure is only used by WinFSP (and not exposed to the user) which // is deep in C++ land where Rust safety rules do not apply. -unsafe impl Send for FileSystem {} +unsafe impl Send for FileSystem {} -impl FileSystem { +impl FileSystem { pub fn volume_params(&self) -> &VolumeParams { &self.params.volume_params } @@ -353,7 +353,7 @@ impl FileSystem { ) -> Result { unsafe { let mut p_inner = std::ptr::null_mut(); - let interface = Box::into_raw(Box::new(Interface::interface::())); + let interface = Box::into_raw(Box::new(TrampolineInterface::interface::())); params .volume_params diff --git a/winfsp_wrs/src/lib.rs b/winfsp_wrs/src/lib.rs index 9b31eda..49a1b93 100644 --- a/winfsp_wrs/src/lib.rs +++ b/winfsp_wrs/src/lib.rs @@ -6,9 +6,9 @@ mod info; mod init; mod security; -pub(crate) use callback::Interface; +pub(crate) use callback::TrampolineInterface; -pub use callback::{FileContextKind, FileSystemContext}; +pub use callback::{FileContextKind, FileSystemInterface}; #[cfg(feature = "icon")] pub use file_system::set_folder_icon; pub use file_system::{