2014-06-23 18:59:45 +04:00
/*
* OS X and Netatalk interoperability VFS module for Samba - 3. x
*
* Copyright ( C ) Ralph Boehme , 2013 , 2014
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation ; either version 3 of the License , or
* ( at your option ) any later version .
*
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
*
* You should have received a copy of the GNU General Public License
* along with this program ; if not , see < http : //www.gnu.org/licenses/>.
*/
# include "includes.h"
# include "MacExtensions.h"
# include "smbd/smbd.h"
# include "system/filesys.h"
# include "lib/util/time.h"
# include "system/shmem.h"
# include "locking/proto.h"
# include "smbd/globals.h"
# include "messages.h"
# include "libcli/security/security.h"
2014-11-26 20:11:17 +03:00
# include "../libcli/smb/smb2_create_ctx.h"
2015-04-22 23:29:16 +03:00
# include "lib/util/tevent_ntstatus.h"
2017-05-12 15:40:03 +03:00
# include "lib/util/tevent_unix.h"
2024-04-10 14:02:39 +03:00
# include "lib/util/util_file.h"
2017-06-03 13:57:59 +03:00
# include "offload_token.h"
2017-10-11 13:58:59 +03:00
# include "string_replace.h"
2019-06-18 17:58:29 +03:00
# include "hash_inode.h"
2019-07-09 14:10:00 +03:00
# include "lib/adouble.h"
2019-07-01 16:53:36 +03:00
# include "lib/util_macstreams.h"
2023-11-27 16:53:07 +03:00
# include "source3/smbd/dir.h"
2018-11-05 19:58:37 +03:00
2014-06-23 18:59:45 +04:00
/*
* Enhanced OS X and Netatalk compatibility
* = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = =
*
* This modules takes advantage of vfs_streams_xattr and
* vfs_catia . VFS modules vfs_fruit and vfs_streams_xattr must be
* loaded in the correct order :
*
* vfs modules = catia fruit streams_xattr
*
* The module intercepts the OS X special streams " AFP_AfpInfo " and
* " AFP_Resource " and handles them in a special way . All other named
* streams are deferred to vfs_streams_xattr .
*
* The OS X client maps all NTFS illegal characters to the Unicode
2019-07-04 14:55:28 +03:00
* private range . This module optionally stores the characters using
2014-06-23 18:59:45 +04:00
* their native ASCII encoding using vfs_catia . If you ' re not enabling
* this feature , you can skip catia from vfs modules .
*
* Finally , open modes are optionally checked against Netatalk AFP
* share modes .
*
* The " AFP_AfpInfo " named stream is a binary blob containing OS X
* extended metadata for files and directories . This module optionally
* reads and stores this metadata in a way compatible with Netatalk 3
* which stores the metadata in an EA " org.netatalk.metadata " . Cf
* source3 / include / MacExtensions . h for a description of the binary
* blobs content .
*
* The " AFP_Resource " named stream may be arbitrarily large , thus it
* can ' t be stored in an xattr on most filesystem . ZFS on Solaris is
* the only available filesystem where xattrs can be of any size and
* the OS supports using the file APIs for xattrs .
*
* The AFP_Resource stream is stored in an AppleDouble file prepending
* " ._ " to the filename . On Solaris with ZFS the stream is optionally
2016-11-08 14:35:12 +03:00
* stored in an EA " org.netatalk.resource " .
2014-06-23 18:59:45 +04:00
*
*
* Extended Attributes
* = = = = = = = = = = = = = = = = = = =
*
* The OS X SMB client sends xattrs as ADS too . For xattr interop with
* other protocols you may want to adjust the xattr names the VFS
* module vfs_streams_xattr uses for storing ADS ' s . This defaults to
* user . DosStream . ADS_NAME : $ DATA and can be changed by specifying
* these module parameters :
*
* streams_xattr : prefix = user .
* streams_xattr : store_stream_type = false
*
*
* TODO
* = = = =
*
* - log diagnostic if any needed VFS module is not loaded
* ( eg with lp_vfs_objects ( ) )
* - add tests
*/
static int vfs_fruit_debug_level = DBGC_VFS ;
2017-02-28 11:39:37 +03:00
static struct global_fruit_config {
bool nego_aapl ; /* client negotiated AAPL */
} global_fruit_config ;
2014-06-23 18:59:45 +04:00
# undef DBGC_CLASS
# define DBGC_CLASS vfs_fruit_debug_level
# define FRUIT_PARAM_TYPE_NAME "fruit"
2019-07-09 14:10:00 +03:00
enum apple_fork { APPLE_FORK_DATA , APPLE_FORK_RSRC } ;
2014-06-23 18:59:45 +04:00
2019-07-09 14:10:00 +03:00
enum fruit_rsrc { FRUIT_RSRC_STREAM , FRUIT_RSRC_ADFILE , FRUIT_RSRC_XATTR } ;
enum fruit_meta { FRUIT_META_STREAM , FRUIT_META_NETATALK } ;
enum fruit_locking { FRUIT_LOCKING_NETATALK , FRUIT_LOCKING_NONE } ;
enum fruit_encoding { FRUIT_ENC_NATIVE , FRUIT_ENC_PRIVATE } ;
2016-12-08 21:12:32 +03:00
2019-07-09 14:10:00 +03:00
struct fruit_config_data {
enum fruit_rsrc rsrc ;
enum fruit_meta meta ;
enum fruit_locking locking ;
enum fruit_encoding encoding ;
bool use_aapl ; /* config from smb.conf */
bool use_copyfile ;
bool readdir_attr_enabled ;
bool unix_info_enabled ;
bool copyfile_enabled ;
bool veto_appledouble ;
bool aapl_zero_file_id ;
const char * model ;
bool time_machine ;
off_t time_machine_max_size ;
2023-05-22 13:32:00 +03:00
bool convert_adouble ;
2019-07-09 14:10:00 +03:00
bool wipe_intentionally_left_blank_rfork ;
bool delete_empty_adfiles ;
2023-07-04 18:46:40 +03:00
bool validate_afpinfo ;
2014-06-23 18:59:45 +04:00
2019-07-09 14:10:00 +03:00
/*
* Additional options , all enabled by default ,
* possibly useful for analyzing performance . The associated
* operations with each of them may be expensive , so having
* the chance to disable them individually gives a chance
* tweaking the setup for the particular usecase .
*/
bool readdir_attr_rsize ;
bool readdir_attr_finder_info ;
bool readdir_attr_max_access ;
2021-07-02 23:40:39 +03:00
/* Recursion guard. Will go away when we have STATX. */
bool in_openat_pathref_fsp ;
2019-07-09 14:10:00 +03:00
} ;
2016-12-08 21:12:32 +03:00
2019-07-09 14:10:00 +03:00
static const struct enum_list fruit_rsrc [ ] = {
{ FRUIT_RSRC_STREAM , " stream " } , /* pass on to vfs_streams_xattr */
{ FRUIT_RSRC_ADFILE , " file " } , /* ._ AppleDouble file */
{ FRUIT_RSRC_XATTR , " xattr " } , /* Netatalk compatible xattr (ZFS only) */
{ - 1 , NULL }
} ;
2014-06-23 18:59:45 +04:00
2019-07-09 14:10:00 +03:00
static const struct enum_list fruit_meta [ ] = {
{ FRUIT_META_STREAM , " stream " } , /* pass on to vfs_streams_xattr */
{ FRUIT_META_NETATALK , " netatalk " } , /* Netatalk compatible xattr */
{ - 1 , NULL }
} ;
2014-06-23 18:59:45 +04:00
2019-07-09 14:10:00 +03:00
static const struct enum_list fruit_locking [ ] = {
{ FRUIT_LOCKING_NETATALK , " netatalk " } , /* synchronize locks with Netatalk */
{ FRUIT_LOCKING_NONE , " none " } ,
{ - 1 , NULL }
} ;
2014-06-23 18:59:45 +04:00
2019-07-09 14:10:00 +03:00
static const struct enum_list fruit_encoding [ ] = {
{ FRUIT_ENC_NATIVE , " native " } , /* map unicode private chars to ASCII */
{ FRUIT_ENC_PRIVATE , " private " } , /* keep unicode private chars */
{ - 1 , NULL }
} ;
2014-06-23 18:59:45 +04:00
2019-07-09 14:10:00 +03:00
struct fio {
2020-12-08 18:29:10 +03:00
vfs_handle_struct * handle ;
files_struct * fsp ; /* backlink to itself */
2019-07-09 14:10:00 +03:00
/* tcon config handle */
struct fruit_config_data * config ;
2014-06-23 18:59:45 +04:00
2020-12-08 18:29:10 +03:00
/* Backend fsp for AppleDouble file, can be NULL */
files_struct * ad_fsp ;
/* link from adouble_open_from_base_fsp() to fio */
struct fio * real_fio ;
2019-07-09 14:10:00 +03:00
/* Denote stream type, meta or rsrc */
adouble_type_t type ;
2014-06-23 18:59:45 +04:00
2019-07-09 14:10:00 +03:00
/*
* AFP_AfpInfo stream created , but not written yet , thus still a fake
* pipe fd . This is set to true in fruit_open_meta if there was no
2019-08-29 22:50:56 +03:00
* existing stream but the caller requested O_CREAT . It is later set to
2019-07-09 14:10:00 +03:00
* false when we get a write on the stream that then does open and
* create the stream .
*/
bool fake_fd ;
int flags ;
2016-12-08 21:12:32 +03:00
int mode ;
2019-07-09 14:10:00 +03:00
} ;
2014-06-23 18:59:45 +04:00
/*****************************************************************************
* Helper functions
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2021-07-02 23:43:49 +03:00
static struct adouble * ad_get_meta_fsp ( TALLOC_CTX * ctx ,
vfs_handle_struct * handle ,
const struct smb_filename * smb_fname )
{
NTSTATUS status ;
struct adouble * ad = NULL ;
struct smb_filename * smb_fname_cp = NULL ;
struct fruit_config_data * config = NULL ;
if ( smb_fname - > fsp ! = NULL ) {
return ad_get ( ctx , handle , smb_fname , ADOUBLE_META ) ;
}
SMB_VFS_HANDLE_GET_DATA ( handle ,
config ,
struct fruit_config_data ,
return NULL ) ;
if ( config - > in_openat_pathref_fsp ) {
return NULL ;
}
smb_fname_cp = cp_smb_filename ( ctx ,
smb_fname ) ;
if ( smb_fname_cp = = NULL ) {
return NULL ;
}
2021-07-13 21:40:09 +03:00
TALLOC_FREE ( smb_fname_cp - > stream_name ) ;
2021-07-02 23:43:49 +03:00
config - > in_openat_pathref_fsp = true ;
status = openat_pathref_fsp ( handle - > conn - > cwd_fsp ,
smb_fname_cp ) ;
config - > in_openat_pathref_fsp = false ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
TALLOC_FREE ( smb_fname_cp ) ;
return NULL ;
}
ad = ad_get ( ctx , handle , smb_fname_cp , ADOUBLE_META ) ;
TALLOC_FREE ( smb_fname_cp ) ;
return ad ;
}
2020-12-10 15:11:06 +03:00
static struct fio * fruit_get_complete_fio ( vfs_handle_struct * handle ,
files_struct * fsp )
{
struct fio * fio = ( struct fio * ) VFS_FETCH_FSP_EXTENSION ( handle , fsp ) ;
if ( fio = = NULL ) {
return NULL ;
}
2020-12-08 18:29:10 +03:00
if ( fio - > real_fio ! = NULL ) {
/*
* This is an fsp from adouble_open_from_base_fsp ( )
* we should just pass this to the next
* module .
*/
return NULL ;
}
2020-12-10 15:11:06 +03:00
return fio ;
}
2014-06-23 18:59:45 +04:00
/**
* Initialize config struct from our smb . conf config parameters
* */
static int init_fruit_config ( vfs_handle_struct * handle )
{
struct fruit_config_data * config ;
2023-07-13 10:03:18 +03:00
int enumval = - 1 ;
2017-11-03 12:56:29 +03:00
const char * tm_size_str = NULL ;
2014-06-23 18:59:45 +04:00
config = talloc_zero ( handle - > conn , struct fruit_config_data ) ;
if ( ! config ) {
DEBUG ( 1 , ( " talloc_zero() failed \n " ) ) ;
errno = ENOMEM ;
return - 1 ;
}
2016-11-08 14:35:12 +03:00
enumval = lp_parm_enum ( SNUM ( handle - > conn ) , FRUIT_PARAM_TYPE_NAME ,
2023-07-13 10:03:18 +03:00
" resource " , fruit_rsrc , FRUIT_RSRC_ADFILE ) ;
2016-11-08 14:35:12 +03:00
if ( enumval = = - 1 ) {
DEBUG ( 1 , ( " value for %s: resource type unknown \n " ,
2014-06-23 18:59:45 +04:00
FRUIT_PARAM_TYPE_NAME ) ) ;
return - 1 ;
}
config - > rsrc = ( enum fruit_rsrc ) enumval ;
enumval = lp_parm_enum ( SNUM ( handle - > conn ) , FRUIT_PARAM_TYPE_NAME ,
" metadata " , fruit_meta , FRUIT_META_NETATALK ) ;
if ( enumval = = - 1 ) {
DEBUG ( 1 , ( " value for %s: metadata type unknown \n " ,
FRUIT_PARAM_TYPE_NAME ) ) ;
return - 1 ;
}
config - > meta = ( enum fruit_meta ) enumval ;
enumval = lp_parm_enum ( SNUM ( handle - > conn ) , FRUIT_PARAM_TYPE_NAME ,
" locking " , fruit_locking , FRUIT_LOCKING_NONE ) ;
if ( enumval = = - 1 ) {
DEBUG ( 1 , ( " value for %s: locking type unknown \n " ,
FRUIT_PARAM_TYPE_NAME ) ) ;
return - 1 ;
}
config - > locking = ( enum fruit_locking ) enumval ;
enumval = lp_parm_enum ( SNUM ( handle - > conn ) , FRUIT_PARAM_TYPE_NAME ,
" encoding " , fruit_encoding , FRUIT_ENC_PRIVATE ) ;
if ( enumval = = - 1 ) {
DEBUG ( 1 , ( " value for %s: encoding type unknown \n " ,
FRUIT_PARAM_TYPE_NAME ) ) ;
return - 1 ;
}
config - > encoding = ( enum fruit_encoding ) enumval ;
2017-01-19 11:30:45 +03:00
if ( config - > rsrc = = FRUIT_RSRC_ADFILE ) {
config - > veto_appledouble = lp_parm_bool ( SNUM ( handle - > conn ) ,
FRUIT_PARAM_TYPE_NAME ,
" veto_appledouble " ,
true ) ;
}
2015-05-09 09:31:24 +03:00
2015-06-15 19:31:23 +03:00
config - > use_aapl = lp_parm_bool (
- 1 , FRUIT_PARAM_TYPE_NAME , " aapl " , true ) ;
2014-11-26 20:11:17 +03:00
2016-11-14 21:14:44 +03:00
config - > time_machine = lp_parm_bool (
SNUM ( handle - > conn ) , FRUIT_PARAM_TYPE_NAME , " time machine " , false ) ;
2015-06-15 19:31:23 +03:00
config - > unix_info_enabled = lp_parm_bool (
- 1 , FRUIT_PARAM_TYPE_NAME , " nfs_aces " , true ) ;
2015-03-25 17:09:02 +03:00
2015-04-22 23:29:16 +03:00
config - > use_copyfile = lp_parm_bool ( - 1 , FRUIT_PARAM_TYPE_NAME ,
" copyfile " , false ) ;
2017-03-23 15:08:45 +03:00
config - > aapl_zero_file_id =
2019-06-30 15:24:59 +03:00
lp_parm_bool ( SNUM ( handle - > conn ) , FRUIT_PARAM_TYPE_NAME ,
2022-03-29 18:38:15 +03:00
" zero_file_id " , true ) ;
2017-03-23 15:08:45 +03:00
2015-06-15 19:31:23 +03:00
config - > readdir_attr_rsize = lp_parm_bool (
SNUM ( handle - > conn ) , " readdir_attr " , " aapl_rsize " , true ) ;
2014-11-26 20:11:17 +03:00
2015-06-15 19:31:23 +03:00
config - > readdir_attr_finder_info = lp_parm_bool (
SNUM ( handle - > conn ) , " readdir_attr " , " aapl_finder_info " , true ) ;
2014-11-26 20:11:17 +03:00
2015-06-15 19:31:23 +03:00
config - > readdir_attr_max_access = lp_parm_bool (
SNUM ( handle - > conn ) , " readdir_attr " , " aapl_max_access " , true ) ;
2014-11-26 20:11:17 +03:00
2017-06-28 19:10:28 +03:00
config - > model = lp_parm_const_string (
- 1 , FRUIT_PARAM_TYPE_NAME , " model " , " MacSamba " ) ;
2017-11-03 12:56:29 +03:00
tm_size_str = lp_parm_const_string (
SNUM ( handle - > conn ) , FRUIT_PARAM_TYPE_NAME ,
" time machine max size " , NULL ) ;
if ( tm_size_str ! = NULL ) {
2018-02-22 17:52:46 +03:00
config - > time_machine_max_size = conv_str_size ( tm_size_str ) ;
2017-11-03 12:56:29 +03:00
}
2023-05-22 13:32:00 +03:00
config - > convert_adouble = lp_parm_bool (
SNUM ( handle - > conn ) , FRUIT_PARAM_TYPE_NAME ,
" convert_adouble " , true ) ;
2018-10-03 13:01:00 +03:00
config - > wipe_intentionally_left_blank_rfork = lp_parm_bool (
SNUM ( handle - > conn ) , FRUIT_PARAM_TYPE_NAME ,
" wipe_intentionally_left_blank_rfork " , false ) ;
2018-10-03 13:01:00 +03:00
config - > delete_empty_adfiles = lp_parm_bool (
SNUM ( handle - > conn ) , FRUIT_PARAM_TYPE_NAME ,
" delete_empty_adfiles " , false ) ;
2023-07-04 18:46:40 +03:00
config - > validate_afpinfo = lp_parm_bool (
SNUM ( handle - > conn ) , FRUIT_PARAM_TYPE_NAME ,
" validate_afpinfo " , true ) ;
2014-06-23 18:59:45 +04:00
SMB_VFS_HANDLE_SET_DATA ( handle , config ,
NULL , struct fruit_config_data ,
return - 1 ) ;
return 0 ;
}
static bool add_fruit_stream ( TALLOC_CTX * mem_ctx , unsigned int * num_streams ,
struct stream_struct * * streams ,
const char * name , off_t size ,
off_t alloc_size )
{
struct stream_struct * tmp ;
tmp = talloc_realloc ( mem_ctx , * streams , struct stream_struct ,
( * num_streams ) + 1 ) ;
if ( tmp = = NULL ) {
return false ;
}
tmp [ * num_streams ] . name = talloc_asprintf ( tmp , " %s:$DATA " , name ) ;
if ( tmp [ * num_streams ] . name = = NULL ) {
return false ;
}
tmp [ * num_streams ] . size = size ;
tmp [ * num_streams ] . alloc_size = alloc_size ;
* streams = tmp ;
* num_streams + = 1 ;
return true ;
}
2024-11-04 20:17:16 +03:00
static void filter_empty_rsrc_stream ( unsigned int * num_streams ,
2016-12-02 17:49:03 +03:00
struct stream_struct * * streams )
{
unsigned int i ;
for ( i = 0 ; i < * num_streams ; i + + ) {
2024-11-04 20:17:16 +03:00
struct stream_struct * s = & ( * streams ) [ i ] ;
if ( strequal_m ( s - > name , AFPRESOURCE_STREAM ) & &
( s - > size = = 0 ) ) {
TALLOC_FREE ( s - > name ) ;
ARRAY_DEL_ELEMENT ( streams , i , * num_streams ) ;
* num_streams - = 1 ;
return ;
2016-12-02 17:49:03 +03:00
}
}
}
2015-08-24 18:43:40 +03:00
static bool del_fruit_stream ( TALLOC_CTX * mem_ctx , unsigned int * num_streams ,
struct stream_struct * * streams ,
const char * name )
{
struct stream_struct * tmp = * streams ;
2016-05-19 19:42:04 +03:00
unsigned int i ;
2015-08-24 18:43:40 +03:00
if ( * num_streams = = 0 ) {
return true ;
}
for ( i = 0 ; i < * num_streams ; i + + ) {
if ( strequal_m ( tmp [ i ] . name , name ) ) {
break ;
}
}
if ( i = = * num_streams ) {
return true ;
}
TALLOC_FREE ( tmp [ i ] . name ) ;
2020-03-26 12:41:30 +03:00
ARRAY_DEL_ELEMENT ( tmp , i , * num_streams ) ;
2015-08-24 18:43:40 +03:00
* num_streams - = 1 ;
return true ;
}
2016-12-09 18:25:38 +03:00
static bool ad_empty_finderinfo ( const struct adouble * ad )
2014-06-23 18:59:45 +04:00
{
2016-12-09 18:25:38 +03:00
int cmp ;
2014-06-23 18:59:45 +04:00
char emptybuf [ ADEDLEN_FINDERI ] = { 0 } ;
2016-12-09 18:25:38 +03:00
char * fi = NULL ;
2016-11-16 13:01:45 +03:00
fi = ad_get_entry ( ad , ADEID_FINDERI ) ;
2016-12-09 18:25:38 +03:00
if ( fi = = NULL ) {
DBG_ERR ( " Missing FinderInfo in struct adouble [%p] \n " , ad ) ;
return false ;
2014-06-23 18:59:45 +04:00
}
2016-12-09 18:25:38 +03:00
cmp = memcmp ( emptybuf , fi , ADEDLEN_FINDERI ) ;
return ( cmp = = 0 ) ;
2014-06-23 18:59:45 +04:00
}
2016-11-15 23:32:25 +03:00
static bool ai_empty_finderinfo ( const AfpInfo * ai )
{
int cmp ;
char emptybuf [ ADEDLEN_FINDERI ] = { 0 } ;
cmp = memcmp ( emptybuf , & ai - > afpi_FinderInfo [ 0 ] , ADEDLEN_FINDERI ) ;
return ( cmp = = 0 ) ;
}
2014-06-23 18:59:45 +04:00
/**
* Update btime with btime from Netatalk
* */
static void update_btime ( vfs_handle_struct * handle ,
struct smb_filename * smb_fname )
{
uint32_t t ;
struct timespec creation_time = { 0 } ;
struct adouble * ad ;
2016-12-02 19:25:47 +03:00
struct fruit_config_data * config = NULL ;
SMB_VFS_HANDLE_GET_DATA ( handle , config , struct fruit_config_data ,
return ) ;
switch ( config - > meta ) {
case FRUIT_META_STREAM :
return ;
case FRUIT_META_NETATALK :
/* Handled below */
break ;
default :
DBG_ERR ( " Unexpected meta config [%d] \n " , config - > meta ) ;
return ;
}
2014-06-23 18:59:45 +04:00
2021-07-02 23:50:48 +03:00
ad = ad_get_meta_fsp ( talloc_tos ( ) , handle , smb_fname ) ;
2014-06-23 18:59:45 +04:00
if ( ad = = NULL ) {
return ;
}
if ( ad_getdate ( ad , AD_DATE_UNIX | AD_DATE_CREATE , & t ) ! = 0 ) {
TALLOC_FREE ( ad ) ;
return ;
}
TALLOC_FREE ( ad ) ;
creation_time . tv_sec = convert_uint32_t_to_time_t ( t ) ;
update_stat_ex_create_time ( & smb_fname - > st , creation_time ) ;
return ;
}
/**
* Map an access mask to a Netatalk single byte byte range lock
* */
2014-11-13 11:12:56 +03:00
static off_t access_to_netatalk_brl ( enum apple_fork fork_type ,
2014-06-23 18:59:45 +04:00
uint32_t access_mask )
{
off_t offset ;
switch ( access_mask ) {
case FILE_READ_DATA :
offset = AD_FILELOCK_OPEN_RD ;
break ;
case FILE_WRITE_DATA :
case FILE_APPEND_DATA :
offset = AD_FILELOCK_OPEN_WR ;
break ;
default :
offset = AD_FILELOCK_OPEN_NONE ;
break ;
}
2014-11-13 11:12:56 +03:00
if ( fork_type = = APPLE_FORK_RSRC ) {
2014-06-23 18:59:45 +04:00
if ( offset = = AD_FILELOCK_OPEN_NONE ) {
offset = AD_FILELOCK_RSRC_OPEN_NONE ;
} else {
offset + = 2 ;
}
}
return offset ;
}
/**
* Map a deny mode to a Netatalk brl
* */
2014-11-13 11:12:56 +03:00
static off_t denymode_to_netatalk_brl ( enum apple_fork fork_type ,
2014-06-23 18:59:45 +04:00
uint32_t deny_mode )
{
2018-09-05 12:31:10 +03:00
off_t offset = 0 ;
2014-06-23 18:59:45 +04:00
switch ( deny_mode ) {
case DENY_READ :
offset = AD_FILELOCK_DENY_RD ;
break ;
case DENY_WRITE :
offset = AD_FILELOCK_DENY_WR ;
break ;
default :
smb_panic ( " denymode_to_netatalk_brl: bad deny mode \n " ) ;
}
2014-11-13 11:12:56 +03:00
if ( fork_type = = APPLE_FORK_RSRC ) {
2014-06-23 18:59:45 +04:00
offset + = 2 ;
}
return offset ;
}
/**
* Call fcntl ( ) with an exclusive F_GETLK request in order to
2019-08-29 22:50:56 +03:00
* determine if there ' s an existing shared lock
2014-06-23 18:59:45 +04:00
*
2017-02-17 22:46:28 +03:00
* @ return true if the requested lock was found or any error occurred
2014-06-23 18:59:45 +04:00
* false if the lock was not found
* */
static bool test_netatalk_lock ( files_struct * fsp , off_t in_offset )
{
bool result ;
off_t offset = in_offset ;
off_t len = 1 ;
int type = F_WRLCK ;
2019-01-30 20:09:52 +03:00
pid_t pid = 0 ;
2014-06-23 18:59:45 +04:00
result = SMB_VFS_GETLOCK ( fsp , & offset , & len , & type , & pid ) ;
if ( result = = false ) {
return true ;
}
if ( type ! = F_UNLCK ) {
return true ;
}
return false ;
}
static NTSTATUS fruit_check_access ( vfs_handle_struct * handle ,
files_struct * fsp ,
uint32_t access_mask ,
s3: VFS: vfs_fruit. Fix the NetAtalk deny mode compatibility code.
This exhibited itself as a problem with OFD locks reported
as:
BUG: https://bugzilla.samba.org/show_bug.cgi?id=13770
However, due to underlying bugs in the vfs_fruit
code the file locks were not being properly applied.
There are two problems in fruit_check_access().
Problem #1:
Inside fruit_check_access() we have:
flags = fcntl(fsp->fh->fd, F_GETFL);
..
if (flags & (O_RDONLY|O_RDWR)) {
We shouldn't be calling fcntl(fsp->fh->fd, ..) directly.
fsp->fh->fd may be a made up number from an underlying
VFS module that has no meaning to a system call.
Secondly, in all POSIX systems - O_RDONLY is defined as
*zero*. O_RDWR = 2.
Which means flags & (O_RDONLY|O_RDWR) becomes (flags & 2),
not what we actually thought.
Problem #2:
deny_mode is *not* a bitmask, it's a set of discrete values.
Inside fruit_check_access() we have:
if (deny_mode & DENY_READ) and also (deny_mode & DENY_WRITE)
However, deny modes are defined as:
/* deny modes */
define DENY_DOS 0
define DENY_ALL 1
define DENY_WRITE 2
define DENY_READ 3
define DENY_NONE 4
define DENY_FCB 7
so if deny_mode = DENY_WRITE, or if deny_mode = DENY_READ
then it's going to trigger both the if (deny_mode & DENY_READ)
*and* the (deny_mode & DENY_WRITE) conditions.
These problems allowed the original test test_netatalk_lock code to
pass (which was added for BUG: https://bugzilla.samba.org/show_bug.cgi?id=13584
to demonstrate the lock order violation).
This patch refactors the fruit_check_access()
code to be much simpler (IMHO) to understand.
Firstly, pass in the SMB1/2 share mode, not old
DOS deny modes.
Secondly, read all the possible NetAtalk locks
into local variables:
netatalk_already_open_for_reading
netatalk_already_open_with_deny_read
netatalk_already_open_for_writing
netatalk_already_open_with_deny_write
Then do the share mode/access mode checks
with the requested values against any stored
netatalk modes/access modes.
Finally add in NetATalk compatible locks
that represent our share modes/access modes
into the file, with an early return if we don't
have FILE_READ_DATA (in which case we can't
write locks anyway).
The patch is easier to understand by looking
at the completed patched fruit_check_access()
function, rather than trying to look at the
diff.
Signed-off-by: Jeremy Allison <jra@samba.org>
Reviewed-by: Ralph Böhme <slow@samba.org>
2019-02-07 04:49:16 +03:00
uint32_t share_mode )
2014-06-23 18:59:45 +04:00
{
NTSTATUS status = NT_STATUS_OK ;
off_t off ;
s3: VFS: vfs_fruit. Fix the NetAtalk deny mode compatibility code.
This exhibited itself as a problem with OFD locks reported
as:
BUG: https://bugzilla.samba.org/show_bug.cgi?id=13770
However, due to underlying bugs in the vfs_fruit
code the file locks were not being properly applied.
There are two problems in fruit_check_access().
Problem #1:
Inside fruit_check_access() we have:
flags = fcntl(fsp->fh->fd, F_GETFL);
..
if (flags & (O_RDONLY|O_RDWR)) {
We shouldn't be calling fcntl(fsp->fh->fd, ..) directly.
fsp->fh->fd may be a made up number from an underlying
VFS module that has no meaning to a system call.
Secondly, in all POSIX systems - O_RDONLY is defined as
*zero*. O_RDWR = 2.
Which means flags & (O_RDONLY|O_RDWR) becomes (flags & 2),
not what we actually thought.
Problem #2:
deny_mode is *not* a bitmask, it's a set of discrete values.
Inside fruit_check_access() we have:
if (deny_mode & DENY_READ) and also (deny_mode & DENY_WRITE)
However, deny modes are defined as:
/* deny modes */
define DENY_DOS 0
define DENY_ALL 1
define DENY_WRITE 2
define DENY_READ 3
define DENY_NONE 4
define DENY_FCB 7
so if deny_mode = DENY_WRITE, or if deny_mode = DENY_READ
then it's going to trigger both the if (deny_mode & DENY_READ)
*and* the (deny_mode & DENY_WRITE) conditions.
These problems allowed the original test test_netatalk_lock code to
pass (which was added for BUG: https://bugzilla.samba.org/show_bug.cgi?id=13584
to demonstrate the lock order violation).
This patch refactors the fruit_check_access()
code to be much simpler (IMHO) to understand.
Firstly, pass in the SMB1/2 share mode, not old
DOS deny modes.
Secondly, read all the possible NetAtalk locks
into local variables:
netatalk_already_open_for_reading
netatalk_already_open_with_deny_read
netatalk_already_open_for_writing
netatalk_already_open_with_deny_write
Then do the share mode/access mode checks
with the requested values against any stored
netatalk modes/access modes.
Finally add in NetATalk compatible locks
that represent our share modes/access modes
into the file, with an early return if we don't
have FILE_READ_DATA (in which case we can't
write locks anyway).
The patch is easier to understand by looking
at the completed patched fruit_check_access()
function, rather than trying to look at the
diff.
Signed-off-by: Jeremy Allison <jra@samba.org>
Reviewed-by: Ralph Böhme <slow@samba.org>
2019-02-07 04:49:16 +03:00
bool share_for_read = ( share_mode & FILE_SHARE_READ ) ;
bool share_for_write = ( share_mode & FILE_SHARE_WRITE ) ;
bool netatalk_already_open_for_reading = false ;
bool netatalk_already_open_for_writing = false ;
bool netatalk_already_open_with_deny_read = false ;
bool netatalk_already_open_with_deny_write = false ;
2019-08-08 20:26:28 +03:00
struct GUID req_guid = GUID_random ( ) ;
2014-06-23 18:59:45 +04:00
/* FIXME: hardcoded data fork, add resource fork */
2014-11-13 11:12:56 +03:00
enum apple_fork fork_type = APPLE_FORK_DATA ;
2014-06-23 18:59:45 +04:00
s3: VFS: vfs_fruit. Fix the NetAtalk deny mode compatibility code.
This exhibited itself as a problem with OFD locks reported
as:
BUG: https://bugzilla.samba.org/show_bug.cgi?id=13770
However, due to underlying bugs in the vfs_fruit
code the file locks were not being properly applied.
There are two problems in fruit_check_access().
Problem #1:
Inside fruit_check_access() we have:
flags = fcntl(fsp->fh->fd, F_GETFL);
..
if (flags & (O_RDONLY|O_RDWR)) {
We shouldn't be calling fcntl(fsp->fh->fd, ..) directly.
fsp->fh->fd may be a made up number from an underlying
VFS module that has no meaning to a system call.
Secondly, in all POSIX systems - O_RDONLY is defined as
*zero*. O_RDWR = 2.
Which means flags & (O_RDONLY|O_RDWR) becomes (flags & 2),
not what we actually thought.
Problem #2:
deny_mode is *not* a bitmask, it's a set of discrete values.
Inside fruit_check_access() we have:
if (deny_mode & DENY_READ) and also (deny_mode & DENY_WRITE)
However, deny modes are defined as:
/* deny modes */
define DENY_DOS 0
define DENY_ALL 1
define DENY_WRITE 2
define DENY_READ 3
define DENY_NONE 4
define DENY_FCB 7
so if deny_mode = DENY_WRITE, or if deny_mode = DENY_READ
then it's going to trigger both the if (deny_mode & DENY_READ)
*and* the (deny_mode & DENY_WRITE) conditions.
These problems allowed the original test test_netatalk_lock code to
pass (which was added for BUG: https://bugzilla.samba.org/show_bug.cgi?id=13584
to demonstrate the lock order violation).
This patch refactors the fruit_check_access()
code to be much simpler (IMHO) to understand.
Firstly, pass in the SMB1/2 share mode, not old
DOS deny modes.
Secondly, read all the possible NetAtalk locks
into local variables:
netatalk_already_open_for_reading
netatalk_already_open_with_deny_read
netatalk_already_open_for_writing
netatalk_already_open_with_deny_write
Then do the share mode/access mode checks
with the requested values against any stored
netatalk modes/access modes.
Finally add in NetATalk compatible locks
that represent our share modes/access modes
into the file, with an early return if we don't
have FILE_READ_DATA (in which case we can't
write locks anyway).
The patch is easier to understand by looking
at the completed patched fruit_check_access()
function, rather than trying to look at the
diff.
Signed-off-by: Jeremy Allison <jra@samba.org>
Reviewed-by: Ralph Böhme <slow@samba.org>
2019-02-07 04:49:16 +03:00
DBG_DEBUG ( " fruit_check_access: %s, am: %s/%s, sm: 0x%x \n " ,
2014-06-23 18:59:45 +04:00
fsp_str_dbg ( fsp ) ,
access_mask & FILE_READ_DATA ? " READ " : " - " ,
access_mask & FILE_WRITE_DATA ? " WRITE " : " - " ,
s3: VFS: vfs_fruit. Fix the NetAtalk deny mode compatibility code.
This exhibited itself as a problem with OFD locks reported
as:
BUG: https://bugzilla.samba.org/show_bug.cgi?id=13770
However, due to underlying bugs in the vfs_fruit
code the file locks were not being properly applied.
There are two problems in fruit_check_access().
Problem #1:
Inside fruit_check_access() we have:
flags = fcntl(fsp->fh->fd, F_GETFL);
..
if (flags & (O_RDONLY|O_RDWR)) {
We shouldn't be calling fcntl(fsp->fh->fd, ..) directly.
fsp->fh->fd may be a made up number from an underlying
VFS module that has no meaning to a system call.
Secondly, in all POSIX systems - O_RDONLY is defined as
*zero*. O_RDWR = 2.
Which means flags & (O_RDONLY|O_RDWR) becomes (flags & 2),
not what we actually thought.
Problem #2:
deny_mode is *not* a bitmask, it's a set of discrete values.
Inside fruit_check_access() we have:
if (deny_mode & DENY_READ) and also (deny_mode & DENY_WRITE)
However, deny modes are defined as:
/* deny modes */
define DENY_DOS 0
define DENY_ALL 1
define DENY_WRITE 2
define DENY_READ 3
define DENY_NONE 4
define DENY_FCB 7
so if deny_mode = DENY_WRITE, or if deny_mode = DENY_READ
then it's going to trigger both the if (deny_mode & DENY_READ)
*and* the (deny_mode & DENY_WRITE) conditions.
These problems allowed the original test test_netatalk_lock code to
pass (which was added for BUG: https://bugzilla.samba.org/show_bug.cgi?id=13584
to demonstrate the lock order violation).
This patch refactors the fruit_check_access()
code to be much simpler (IMHO) to understand.
Firstly, pass in the SMB1/2 share mode, not old
DOS deny modes.
Secondly, read all the possible NetAtalk locks
into local variables:
netatalk_already_open_for_reading
netatalk_already_open_with_deny_read
netatalk_already_open_for_writing
netatalk_already_open_with_deny_write
Then do the share mode/access mode checks
with the requested values against any stored
netatalk modes/access modes.
Finally add in NetATalk compatible locks
that represent our share modes/access modes
into the file, with an early return if we don't
have FILE_READ_DATA (in which case we can't
write locks anyway).
The patch is easier to understand by looking
at the completed patched fruit_check_access()
function, rather than trying to look at the
diff.
Signed-off-by: Jeremy Allison <jra@samba.org>
Reviewed-by: Ralph Böhme <slow@samba.org>
2019-02-07 04:49:16 +03:00
share_mode ) ;
2014-06-23 18:59:45 +04:00
2020-09-26 22:52:52 +03:00
if ( fsp_get_io_fd ( fsp ) = = - 1 ) {
2016-12-08 13:08:53 +03:00
return NT_STATUS_OK ;
}
s3: VFS: vfs_fruit. Fix the NetAtalk deny mode compatibility code.
This exhibited itself as a problem with OFD locks reported
as:
BUG: https://bugzilla.samba.org/show_bug.cgi?id=13770
However, due to underlying bugs in the vfs_fruit
code the file locks were not being properly applied.
There are two problems in fruit_check_access().
Problem #1:
Inside fruit_check_access() we have:
flags = fcntl(fsp->fh->fd, F_GETFL);
..
if (flags & (O_RDONLY|O_RDWR)) {
We shouldn't be calling fcntl(fsp->fh->fd, ..) directly.
fsp->fh->fd may be a made up number from an underlying
VFS module that has no meaning to a system call.
Secondly, in all POSIX systems - O_RDONLY is defined as
*zero*. O_RDWR = 2.
Which means flags & (O_RDONLY|O_RDWR) becomes (flags & 2),
not what we actually thought.
Problem #2:
deny_mode is *not* a bitmask, it's a set of discrete values.
Inside fruit_check_access() we have:
if (deny_mode & DENY_READ) and also (deny_mode & DENY_WRITE)
However, deny modes are defined as:
/* deny modes */
define DENY_DOS 0
define DENY_ALL 1
define DENY_WRITE 2
define DENY_READ 3
define DENY_NONE 4
define DENY_FCB 7
so if deny_mode = DENY_WRITE, or if deny_mode = DENY_READ
then it's going to trigger both the if (deny_mode & DENY_READ)
*and* the (deny_mode & DENY_WRITE) conditions.
These problems allowed the original test test_netatalk_lock code to
pass (which was added for BUG: https://bugzilla.samba.org/show_bug.cgi?id=13584
to demonstrate the lock order violation).
This patch refactors the fruit_check_access()
code to be much simpler (IMHO) to understand.
Firstly, pass in the SMB1/2 share mode, not old
DOS deny modes.
Secondly, read all the possible NetAtalk locks
into local variables:
netatalk_already_open_for_reading
netatalk_already_open_with_deny_read
netatalk_already_open_for_writing
netatalk_already_open_with_deny_write
Then do the share mode/access mode checks
with the requested values against any stored
netatalk modes/access modes.
Finally add in NetATalk compatible locks
that represent our share modes/access modes
into the file, with an early return if we don't
have FILE_READ_DATA (in which case we can't
write locks anyway).
The patch is easier to understand by looking
at the completed patched fruit_check_access()
function, rather than trying to look at the
diff.
Signed-off-by: Jeremy Allison <jra@samba.org>
Reviewed-by: Ralph Böhme <slow@samba.org>
2019-02-07 04:49:16 +03:00
/* Read NetATalk opens and deny modes on the file. */
netatalk_already_open_for_reading = test_netatalk_lock ( fsp ,
access_to_netatalk_brl ( fork_type ,
FILE_READ_DATA ) ) ;
2016-12-08 13:08:53 +03:00
s3: VFS: vfs_fruit. Fix the NetAtalk deny mode compatibility code.
This exhibited itself as a problem with OFD locks reported
as:
BUG: https://bugzilla.samba.org/show_bug.cgi?id=13770
However, due to underlying bugs in the vfs_fruit
code the file locks were not being properly applied.
There are two problems in fruit_check_access().
Problem #1:
Inside fruit_check_access() we have:
flags = fcntl(fsp->fh->fd, F_GETFL);
..
if (flags & (O_RDONLY|O_RDWR)) {
We shouldn't be calling fcntl(fsp->fh->fd, ..) directly.
fsp->fh->fd may be a made up number from an underlying
VFS module that has no meaning to a system call.
Secondly, in all POSIX systems - O_RDONLY is defined as
*zero*. O_RDWR = 2.
Which means flags & (O_RDONLY|O_RDWR) becomes (flags & 2),
not what we actually thought.
Problem #2:
deny_mode is *not* a bitmask, it's a set of discrete values.
Inside fruit_check_access() we have:
if (deny_mode & DENY_READ) and also (deny_mode & DENY_WRITE)
However, deny modes are defined as:
/* deny modes */
define DENY_DOS 0
define DENY_ALL 1
define DENY_WRITE 2
define DENY_READ 3
define DENY_NONE 4
define DENY_FCB 7
so if deny_mode = DENY_WRITE, or if deny_mode = DENY_READ
then it's going to trigger both the if (deny_mode & DENY_READ)
*and* the (deny_mode & DENY_WRITE) conditions.
These problems allowed the original test test_netatalk_lock code to
pass (which was added for BUG: https://bugzilla.samba.org/show_bug.cgi?id=13584
to demonstrate the lock order violation).
This patch refactors the fruit_check_access()
code to be much simpler (IMHO) to understand.
Firstly, pass in the SMB1/2 share mode, not old
DOS deny modes.
Secondly, read all the possible NetAtalk locks
into local variables:
netatalk_already_open_for_reading
netatalk_already_open_with_deny_read
netatalk_already_open_for_writing
netatalk_already_open_with_deny_write
Then do the share mode/access mode checks
with the requested values against any stored
netatalk modes/access modes.
Finally add in NetATalk compatible locks
that represent our share modes/access modes
into the file, with an early return if we don't
have FILE_READ_DATA (in which case we can't
write locks anyway).
The patch is easier to understand by looking
at the completed patched fruit_check_access()
function, rather than trying to look at the
diff.
Signed-off-by: Jeremy Allison <jra@samba.org>
Reviewed-by: Ralph Böhme <slow@samba.org>
2019-02-07 04:49:16 +03:00
netatalk_already_open_with_deny_read = test_netatalk_lock ( fsp ,
denymode_to_netatalk_brl ( fork_type ,
DENY_READ ) ) ;
2014-06-23 18:59:45 +04:00
s3: VFS: vfs_fruit. Fix the NetAtalk deny mode compatibility code.
This exhibited itself as a problem with OFD locks reported
as:
BUG: https://bugzilla.samba.org/show_bug.cgi?id=13770
However, due to underlying bugs in the vfs_fruit
code the file locks were not being properly applied.
There are two problems in fruit_check_access().
Problem #1:
Inside fruit_check_access() we have:
flags = fcntl(fsp->fh->fd, F_GETFL);
..
if (flags & (O_RDONLY|O_RDWR)) {
We shouldn't be calling fcntl(fsp->fh->fd, ..) directly.
fsp->fh->fd may be a made up number from an underlying
VFS module that has no meaning to a system call.
Secondly, in all POSIX systems - O_RDONLY is defined as
*zero*. O_RDWR = 2.
Which means flags & (O_RDONLY|O_RDWR) becomes (flags & 2),
not what we actually thought.
Problem #2:
deny_mode is *not* a bitmask, it's a set of discrete values.
Inside fruit_check_access() we have:
if (deny_mode & DENY_READ) and also (deny_mode & DENY_WRITE)
However, deny modes are defined as:
/* deny modes */
define DENY_DOS 0
define DENY_ALL 1
define DENY_WRITE 2
define DENY_READ 3
define DENY_NONE 4
define DENY_FCB 7
so if deny_mode = DENY_WRITE, or if deny_mode = DENY_READ
then it's going to trigger both the if (deny_mode & DENY_READ)
*and* the (deny_mode & DENY_WRITE) conditions.
These problems allowed the original test test_netatalk_lock code to
pass (which was added for BUG: https://bugzilla.samba.org/show_bug.cgi?id=13584
to demonstrate the lock order violation).
This patch refactors the fruit_check_access()
code to be much simpler (IMHO) to understand.
Firstly, pass in the SMB1/2 share mode, not old
DOS deny modes.
Secondly, read all the possible NetAtalk locks
into local variables:
netatalk_already_open_for_reading
netatalk_already_open_with_deny_read
netatalk_already_open_for_writing
netatalk_already_open_with_deny_write
Then do the share mode/access mode checks
with the requested values against any stored
netatalk modes/access modes.
Finally add in NetATalk compatible locks
that represent our share modes/access modes
into the file, with an early return if we don't
have FILE_READ_DATA (in which case we can't
write locks anyway).
The patch is easier to understand by looking
at the completed patched fruit_check_access()
function, rather than trying to look at the
diff.
Signed-off-by: Jeremy Allison <jra@samba.org>
Reviewed-by: Ralph Böhme <slow@samba.org>
2019-02-07 04:49:16 +03:00
netatalk_already_open_for_writing = test_netatalk_lock ( fsp ,
access_to_netatalk_brl ( fork_type ,
FILE_WRITE_DATA ) ) ;
2014-06-23 18:59:45 +04:00
s3: VFS: vfs_fruit. Fix the NetAtalk deny mode compatibility code.
This exhibited itself as a problem with OFD locks reported
as:
BUG: https://bugzilla.samba.org/show_bug.cgi?id=13770
However, due to underlying bugs in the vfs_fruit
code the file locks were not being properly applied.
There are two problems in fruit_check_access().
Problem #1:
Inside fruit_check_access() we have:
flags = fcntl(fsp->fh->fd, F_GETFL);
..
if (flags & (O_RDONLY|O_RDWR)) {
We shouldn't be calling fcntl(fsp->fh->fd, ..) directly.
fsp->fh->fd may be a made up number from an underlying
VFS module that has no meaning to a system call.
Secondly, in all POSIX systems - O_RDONLY is defined as
*zero*. O_RDWR = 2.
Which means flags & (O_RDONLY|O_RDWR) becomes (flags & 2),
not what we actually thought.
Problem #2:
deny_mode is *not* a bitmask, it's a set of discrete values.
Inside fruit_check_access() we have:
if (deny_mode & DENY_READ) and also (deny_mode & DENY_WRITE)
However, deny modes are defined as:
/* deny modes */
define DENY_DOS 0
define DENY_ALL 1
define DENY_WRITE 2
define DENY_READ 3
define DENY_NONE 4
define DENY_FCB 7
so if deny_mode = DENY_WRITE, or if deny_mode = DENY_READ
then it's going to trigger both the if (deny_mode & DENY_READ)
*and* the (deny_mode & DENY_WRITE) conditions.
These problems allowed the original test test_netatalk_lock code to
pass (which was added for BUG: https://bugzilla.samba.org/show_bug.cgi?id=13584
to demonstrate the lock order violation).
This patch refactors the fruit_check_access()
code to be much simpler (IMHO) to understand.
Firstly, pass in the SMB1/2 share mode, not old
DOS deny modes.
Secondly, read all the possible NetAtalk locks
into local variables:
netatalk_already_open_for_reading
netatalk_already_open_with_deny_read
netatalk_already_open_for_writing
netatalk_already_open_with_deny_write
Then do the share mode/access mode checks
with the requested values against any stored
netatalk modes/access modes.
Finally add in NetATalk compatible locks
that represent our share modes/access modes
into the file, with an early return if we don't
have FILE_READ_DATA (in which case we can't
write locks anyway).
The patch is easier to understand by looking
at the completed patched fruit_check_access()
function, rather than trying to look at the
diff.
Signed-off-by: Jeremy Allison <jra@samba.org>
Reviewed-by: Ralph Böhme <slow@samba.org>
2019-02-07 04:49:16 +03:00
netatalk_already_open_with_deny_write = test_netatalk_lock ( fsp ,
denymode_to_netatalk_brl ( fork_type ,
DENY_WRITE ) ) ;
2014-06-23 18:59:45 +04:00
s3: VFS: vfs_fruit. Fix the NetAtalk deny mode compatibility code.
This exhibited itself as a problem with OFD locks reported
as:
BUG: https://bugzilla.samba.org/show_bug.cgi?id=13770
However, due to underlying bugs in the vfs_fruit
code the file locks were not being properly applied.
There are two problems in fruit_check_access().
Problem #1:
Inside fruit_check_access() we have:
flags = fcntl(fsp->fh->fd, F_GETFL);
..
if (flags & (O_RDONLY|O_RDWR)) {
We shouldn't be calling fcntl(fsp->fh->fd, ..) directly.
fsp->fh->fd may be a made up number from an underlying
VFS module that has no meaning to a system call.
Secondly, in all POSIX systems - O_RDONLY is defined as
*zero*. O_RDWR = 2.
Which means flags & (O_RDONLY|O_RDWR) becomes (flags & 2),
not what we actually thought.
Problem #2:
deny_mode is *not* a bitmask, it's a set of discrete values.
Inside fruit_check_access() we have:
if (deny_mode & DENY_READ) and also (deny_mode & DENY_WRITE)
However, deny modes are defined as:
/* deny modes */
define DENY_DOS 0
define DENY_ALL 1
define DENY_WRITE 2
define DENY_READ 3
define DENY_NONE 4
define DENY_FCB 7
so if deny_mode = DENY_WRITE, or if deny_mode = DENY_READ
then it's going to trigger both the if (deny_mode & DENY_READ)
*and* the (deny_mode & DENY_WRITE) conditions.
These problems allowed the original test test_netatalk_lock code to
pass (which was added for BUG: https://bugzilla.samba.org/show_bug.cgi?id=13584
to demonstrate the lock order violation).
This patch refactors the fruit_check_access()
code to be much simpler (IMHO) to understand.
Firstly, pass in the SMB1/2 share mode, not old
DOS deny modes.
Secondly, read all the possible NetAtalk locks
into local variables:
netatalk_already_open_for_reading
netatalk_already_open_with_deny_read
netatalk_already_open_for_writing
netatalk_already_open_with_deny_write
Then do the share mode/access mode checks
with the requested values against any stored
netatalk modes/access modes.
Finally add in NetATalk compatible locks
that represent our share modes/access modes
into the file, with an early return if we don't
have FILE_READ_DATA (in which case we can't
write locks anyway).
The patch is easier to understand by looking
at the completed patched fruit_check_access()
function, rather than trying to look at the
diff.
Signed-off-by: Jeremy Allison <jra@samba.org>
Reviewed-by: Ralph Böhme <slow@samba.org>
2019-02-07 04:49:16 +03:00
/* If there are any conflicts - sharing violation. */
if ( ( access_mask & FILE_READ_DATA ) & &
netatalk_already_open_with_deny_read ) {
return NT_STATUS_SHARING_VIOLATION ;
}
2014-06-23 18:59:45 +04:00
s3: VFS: vfs_fruit. Fix the NetAtalk deny mode compatibility code.
This exhibited itself as a problem with OFD locks reported
as:
BUG: https://bugzilla.samba.org/show_bug.cgi?id=13770
However, due to underlying bugs in the vfs_fruit
code the file locks were not being properly applied.
There are two problems in fruit_check_access().
Problem #1:
Inside fruit_check_access() we have:
flags = fcntl(fsp->fh->fd, F_GETFL);
..
if (flags & (O_RDONLY|O_RDWR)) {
We shouldn't be calling fcntl(fsp->fh->fd, ..) directly.
fsp->fh->fd may be a made up number from an underlying
VFS module that has no meaning to a system call.
Secondly, in all POSIX systems - O_RDONLY is defined as
*zero*. O_RDWR = 2.
Which means flags & (O_RDONLY|O_RDWR) becomes (flags & 2),
not what we actually thought.
Problem #2:
deny_mode is *not* a bitmask, it's a set of discrete values.
Inside fruit_check_access() we have:
if (deny_mode & DENY_READ) and also (deny_mode & DENY_WRITE)
However, deny modes are defined as:
/* deny modes */
define DENY_DOS 0
define DENY_ALL 1
define DENY_WRITE 2
define DENY_READ 3
define DENY_NONE 4
define DENY_FCB 7
so if deny_mode = DENY_WRITE, or if deny_mode = DENY_READ
then it's going to trigger both the if (deny_mode & DENY_READ)
*and* the (deny_mode & DENY_WRITE) conditions.
These problems allowed the original test test_netatalk_lock code to
pass (which was added for BUG: https://bugzilla.samba.org/show_bug.cgi?id=13584
to demonstrate the lock order violation).
This patch refactors the fruit_check_access()
code to be much simpler (IMHO) to understand.
Firstly, pass in the SMB1/2 share mode, not old
DOS deny modes.
Secondly, read all the possible NetAtalk locks
into local variables:
netatalk_already_open_for_reading
netatalk_already_open_with_deny_read
netatalk_already_open_for_writing
netatalk_already_open_with_deny_write
Then do the share mode/access mode checks
with the requested values against any stored
netatalk modes/access modes.
Finally add in NetATalk compatible locks
that represent our share modes/access modes
into the file, with an early return if we don't
have FILE_READ_DATA (in which case we can't
write locks anyway).
The patch is easier to understand by looking
at the completed patched fruit_check_access()
function, rather than trying to look at the
diff.
Signed-off-by: Jeremy Allison <jra@samba.org>
Reviewed-by: Ralph Böhme <slow@samba.org>
2019-02-07 04:49:16 +03:00
if ( ! share_for_read & &
netatalk_already_open_for_reading ) {
return NT_STATUS_SHARING_VIOLATION ;
}
2018-08-06 15:33:34 +03:00
s3: VFS: vfs_fruit. Fix the NetAtalk deny mode compatibility code.
This exhibited itself as a problem with OFD locks reported
as:
BUG: https://bugzilla.samba.org/show_bug.cgi?id=13770
However, due to underlying bugs in the vfs_fruit
code the file locks were not being properly applied.
There are two problems in fruit_check_access().
Problem #1:
Inside fruit_check_access() we have:
flags = fcntl(fsp->fh->fd, F_GETFL);
..
if (flags & (O_RDONLY|O_RDWR)) {
We shouldn't be calling fcntl(fsp->fh->fd, ..) directly.
fsp->fh->fd may be a made up number from an underlying
VFS module that has no meaning to a system call.
Secondly, in all POSIX systems - O_RDONLY is defined as
*zero*. O_RDWR = 2.
Which means flags & (O_RDONLY|O_RDWR) becomes (flags & 2),
not what we actually thought.
Problem #2:
deny_mode is *not* a bitmask, it's a set of discrete values.
Inside fruit_check_access() we have:
if (deny_mode & DENY_READ) and also (deny_mode & DENY_WRITE)
However, deny modes are defined as:
/* deny modes */
define DENY_DOS 0
define DENY_ALL 1
define DENY_WRITE 2
define DENY_READ 3
define DENY_NONE 4
define DENY_FCB 7
so if deny_mode = DENY_WRITE, or if deny_mode = DENY_READ
then it's going to trigger both the if (deny_mode & DENY_READ)
*and* the (deny_mode & DENY_WRITE) conditions.
These problems allowed the original test test_netatalk_lock code to
pass (which was added for BUG: https://bugzilla.samba.org/show_bug.cgi?id=13584
to demonstrate the lock order violation).
This patch refactors the fruit_check_access()
code to be much simpler (IMHO) to understand.
Firstly, pass in the SMB1/2 share mode, not old
DOS deny modes.
Secondly, read all the possible NetAtalk locks
into local variables:
netatalk_already_open_for_reading
netatalk_already_open_with_deny_read
netatalk_already_open_for_writing
netatalk_already_open_with_deny_write
Then do the share mode/access mode checks
with the requested values against any stored
netatalk modes/access modes.
Finally add in NetATalk compatible locks
that represent our share modes/access modes
into the file, with an early return if we don't
have FILE_READ_DATA (in which case we can't
write locks anyway).
The patch is easier to understand by looking
at the completed patched fruit_check_access()
function, rather than trying to look at the
diff.
Signed-off-by: Jeremy Allison <jra@samba.org>
Reviewed-by: Ralph Böhme <slow@samba.org>
2019-02-07 04:49:16 +03:00
if ( ( access_mask & FILE_WRITE_DATA ) & &
netatalk_already_open_with_deny_write ) {
return NT_STATUS_SHARING_VIOLATION ;
}
2014-06-23 18:59:45 +04:00
s3: VFS: vfs_fruit. Fix the NetAtalk deny mode compatibility code.
This exhibited itself as a problem with OFD locks reported
as:
BUG: https://bugzilla.samba.org/show_bug.cgi?id=13770
However, due to underlying bugs in the vfs_fruit
code the file locks were not being properly applied.
There are two problems in fruit_check_access().
Problem #1:
Inside fruit_check_access() we have:
flags = fcntl(fsp->fh->fd, F_GETFL);
..
if (flags & (O_RDONLY|O_RDWR)) {
We shouldn't be calling fcntl(fsp->fh->fd, ..) directly.
fsp->fh->fd may be a made up number from an underlying
VFS module that has no meaning to a system call.
Secondly, in all POSIX systems - O_RDONLY is defined as
*zero*. O_RDWR = 2.
Which means flags & (O_RDONLY|O_RDWR) becomes (flags & 2),
not what we actually thought.
Problem #2:
deny_mode is *not* a bitmask, it's a set of discrete values.
Inside fruit_check_access() we have:
if (deny_mode & DENY_READ) and also (deny_mode & DENY_WRITE)
However, deny modes are defined as:
/* deny modes */
define DENY_DOS 0
define DENY_ALL 1
define DENY_WRITE 2
define DENY_READ 3
define DENY_NONE 4
define DENY_FCB 7
so if deny_mode = DENY_WRITE, or if deny_mode = DENY_READ
then it's going to trigger both the if (deny_mode & DENY_READ)
*and* the (deny_mode & DENY_WRITE) conditions.
These problems allowed the original test test_netatalk_lock code to
pass (which was added for BUG: https://bugzilla.samba.org/show_bug.cgi?id=13584
to demonstrate the lock order violation).
This patch refactors the fruit_check_access()
code to be much simpler (IMHO) to understand.
Firstly, pass in the SMB1/2 share mode, not old
DOS deny modes.
Secondly, read all the possible NetAtalk locks
into local variables:
netatalk_already_open_for_reading
netatalk_already_open_with_deny_read
netatalk_already_open_for_writing
netatalk_already_open_with_deny_write
Then do the share mode/access mode checks
with the requested values against any stored
netatalk modes/access modes.
Finally add in NetATalk compatible locks
that represent our share modes/access modes
into the file, with an early return if we don't
have FILE_READ_DATA (in which case we can't
write locks anyway).
The patch is easier to understand by looking
at the completed patched fruit_check_access()
function, rather than trying to look at the
diff.
Signed-off-by: Jeremy Allison <jra@samba.org>
Reviewed-by: Ralph Böhme <slow@samba.org>
2019-02-07 04:49:16 +03:00
if ( ! share_for_write & &
netatalk_already_open_for_writing ) {
return NT_STATUS_SHARING_VIOLATION ;
}
2018-08-06 15:33:34 +03:00
s3: VFS: vfs_fruit. Fix the NetAtalk deny mode compatibility code.
This exhibited itself as a problem with OFD locks reported
as:
BUG: https://bugzilla.samba.org/show_bug.cgi?id=13770
However, due to underlying bugs in the vfs_fruit
code the file locks were not being properly applied.
There are two problems in fruit_check_access().
Problem #1:
Inside fruit_check_access() we have:
flags = fcntl(fsp->fh->fd, F_GETFL);
..
if (flags & (O_RDONLY|O_RDWR)) {
We shouldn't be calling fcntl(fsp->fh->fd, ..) directly.
fsp->fh->fd may be a made up number from an underlying
VFS module that has no meaning to a system call.
Secondly, in all POSIX systems - O_RDONLY is defined as
*zero*. O_RDWR = 2.
Which means flags & (O_RDONLY|O_RDWR) becomes (flags & 2),
not what we actually thought.
Problem #2:
deny_mode is *not* a bitmask, it's a set of discrete values.
Inside fruit_check_access() we have:
if (deny_mode & DENY_READ) and also (deny_mode & DENY_WRITE)
However, deny modes are defined as:
/* deny modes */
define DENY_DOS 0
define DENY_ALL 1
define DENY_WRITE 2
define DENY_READ 3
define DENY_NONE 4
define DENY_FCB 7
so if deny_mode = DENY_WRITE, or if deny_mode = DENY_READ
then it's going to trigger both the if (deny_mode & DENY_READ)
*and* the (deny_mode & DENY_WRITE) conditions.
These problems allowed the original test test_netatalk_lock code to
pass (which was added for BUG: https://bugzilla.samba.org/show_bug.cgi?id=13584
to demonstrate the lock order violation).
This patch refactors the fruit_check_access()
code to be much simpler (IMHO) to understand.
Firstly, pass in the SMB1/2 share mode, not old
DOS deny modes.
Secondly, read all the possible NetAtalk locks
into local variables:
netatalk_already_open_for_reading
netatalk_already_open_with_deny_read
netatalk_already_open_for_writing
netatalk_already_open_with_deny_write
Then do the share mode/access mode checks
with the requested values against any stored
netatalk modes/access modes.
Finally add in NetATalk compatible locks
that represent our share modes/access modes
into the file, with an early return if we don't
have FILE_READ_DATA (in which case we can't
write locks anyway).
The patch is easier to understand by looking
at the completed patched fruit_check_access()
function, rather than trying to look at the
diff.
Signed-off-by: Jeremy Allison <jra@samba.org>
Reviewed-by: Ralph Böhme <slow@samba.org>
2019-02-07 04:49:16 +03:00
if ( ! ( access_mask & FILE_READ_DATA ) ) {
/*
* Nothing we can do here , we need read access
* to set locks .
*/
return NT_STATUS_OK ;
}
2014-06-23 18:59:45 +04:00
s3: VFS: vfs_fruit. Fix the NetAtalk deny mode compatibility code.
This exhibited itself as a problem with OFD locks reported
as:
BUG: https://bugzilla.samba.org/show_bug.cgi?id=13770
However, due to underlying bugs in the vfs_fruit
code the file locks were not being properly applied.
There are two problems in fruit_check_access().
Problem #1:
Inside fruit_check_access() we have:
flags = fcntl(fsp->fh->fd, F_GETFL);
..
if (flags & (O_RDONLY|O_RDWR)) {
We shouldn't be calling fcntl(fsp->fh->fd, ..) directly.
fsp->fh->fd may be a made up number from an underlying
VFS module that has no meaning to a system call.
Secondly, in all POSIX systems - O_RDONLY is defined as
*zero*. O_RDWR = 2.
Which means flags & (O_RDONLY|O_RDWR) becomes (flags & 2),
not what we actually thought.
Problem #2:
deny_mode is *not* a bitmask, it's a set of discrete values.
Inside fruit_check_access() we have:
if (deny_mode & DENY_READ) and also (deny_mode & DENY_WRITE)
However, deny modes are defined as:
/* deny modes */
define DENY_DOS 0
define DENY_ALL 1
define DENY_WRITE 2
define DENY_READ 3
define DENY_NONE 4
define DENY_FCB 7
so if deny_mode = DENY_WRITE, or if deny_mode = DENY_READ
then it's going to trigger both the if (deny_mode & DENY_READ)
*and* the (deny_mode & DENY_WRITE) conditions.
These problems allowed the original test test_netatalk_lock code to
pass (which was added for BUG: https://bugzilla.samba.org/show_bug.cgi?id=13584
to demonstrate the lock order violation).
This patch refactors the fruit_check_access()
code to be much simpler (IMHO) to understand.
Firstly, pass in the SMB1/2 share mode, not old
DOS deny modes.
Secondly, read all the possible NetAtalk locks
into local variables:
netatalk_already_open_for_reading
netatalk_already_open_with_deny_read
netatalk_already_open_for_writing
netatalk_already_open_with_deny_write
Then do the share mode/access mode checks
with the requested values against any stored
netatalk modes/access modes.
Finally add in NetATalk compatible locks
that represent our share modes/access modes
into the file, with an early return if we don't
have FILE_READ_DATA (in which case we can't
write locks anyway).
The patch is easier to understand by looking
at the completed patched fruit_check_access()
function, rather than trying to look at the
diff.
Signed-off-by: Jeremy Allison <jra@samba.org>
Reviewed-by: Ralph Böhme <slow@samba.org>
2019-02-07 04:49:16 +03:00
/* Set NetAtalk locks matching our access */
if ( access_mask & FILE_READ_DATA ) {
off = access_to_netatalk_brl ( fork_type , FILE_READ_DATA ) ;
2019-08-08 20:26:28 +03:00
req_guid . time_hi_and_version = __LINE__ ;
2019-07-01 15:30:15 +03:00
status = do_lock (
2019-06-20 13:20:39 +03:00
fsp ,
2019-08-08 20:26:28 +03:00
talloc_tos ( ) ,
& req_guid ,
2019-06-20 13:20:39 +03:00
fsp - > op - > global - > open_persistent_id ,
1 ,
off ,
READ_LOCK ,
POSIX_LOCK ,
NULL ,
NULL ) ;
2014-06-23 18:59:45 +04:00
s3: VFS: vfs_fruit. Fix the NetAtalk deny mode compatibility code.
This exhibited itself as a problem with OFD locks reported
as:
BUG: https://bugzilla.samba.org/show_bug.cgi?id=13770
However, due to underlying bugs in the vfs_fruit
code the file locks were not being properly applied.
There are two problems in fruit_check_access().
Problem #1:
Inside fruit_check_access() we have:
flags = fcntl(fsp->fh->fd, F_GETFL);
..
if (flags & (O_RDONLY|O_RDWR)) {
We shouldn't be calling fcntl(fsp->fh->fd, ..) directly.
fsp->fh->fd may be a made up number from an underlying
VFS module that has no meaning to a system call.
Secondly, in all POSIX systems - O_RDONLY is defined as
*zero*. O_RDWR = 2.
Which means flags & (O_RDONLY|O_RDWR) becomes (flags & 2),
not what we actually thought.
Problem #2:
deny_mode is *not* a bitmask, it's a set of discrete values.
Inside fruit_check_access() we have:
if (deny_mode & DENY_READ) and also (deny_mode & DENY_WRITE)
However, deny modes are defined as:
/* deny modes */
define DENY_DOS 0
define DENY_ALL 1
define DENY_WRITE 2
define DENY_READ 3
define DENY_NONE 4
define DENY_FCB 7
so if deny_mode = DENY_WRITE, or if deny_mode = DENY_READ
then it's going to trigger both the if (deny_mode & DENY_READ)
*and* the (deny_mode & DENY_WRITE) conditions.
These problems allowed the original test test_netatalk_lock code to
pass (which was added for BUG: https://bugzilla.samba.org/show_bug.cgi?id=13584
to demonstrate the lock order violation).
This patch refactors the fruit_check_access()
code to be much simpler (IMHO) to understand.
Firstly, pass in the SMB1/2 share mode, not old
DOS deny modes.
Secondly, read all the possible NetAtalk locks
into local variables:
netatalk_already_open_for_reading
netatalk_already_open_with_deny_read
netatalk_already_open_for_writing
netatalk_already_open_with_deny_write
Then do the share mode/access mode checks
with the requested values against any stored
netatalk modes/access modes.
Finally add in NetATalk compatible locks
that represent our share modes/access modes
into the file, with an early return if we don't
have FILE_READ_DATA (in which case we can't
write locks anyway).
The patch is easier to understand by looking
at the completed patched fruit_check_access()
function, rather than trying to look at the
diff.
Signed-off-by: Jeremy Allison <jra@samba.org>
Reviewed-by: Ralph Böhme <slow@samba.org>
2019-02-07 04:49:16 +03:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
2014-06-23 18:59:45 +04:00
}
}
s3: VFS: vfs_fruit. Fix the NetAtalk deny mode compatibility code.
This exhibited itself as a problem with OFD locks reported
as:
BUG: https://bugzilla.samba.org/show_bug.cgi?id=13770
However, due to underlying bugs in the vfs_fruit
code the file locks were not being properly applied.
There are two problems in fruit_check_access().
Problem #1:
Inside fruit_check_access() we have:
flags = fcntl(fsp->fh->fd, F_GETFL);
..
if (flags & (O_RDONLY|O_RDWR)) {
We shouldn't be calling fcntl(fsp->fh->fd, ..) directly.
fsp->fh->fd may be a made up number from an underlying
VFS module that has no meaning to a system call.
Secondly, in all POSIX systems - O_RDONLY is defined as
*zero*. O_RDWR = 2.
Which means flags & (O_RDONLY|O_RDWR) becomes (flags & 2),
not what we actually thought.
Problem #2:
deny_mode is *not* a bitmask, it's a set of discrete values.
Inside fruit_check_access() we have:
if (deny_mode & DENY_READ) and also (deny_mode & DENY_WRITE)
However, deny modes are defined as:
/* deny modes */
define DENY_DOS 0
define DENY_ALL 1
define DENY_WRITE 2
define DENY_READ 3
define DENY_NONE 4
define DENY_FCB 7
so if deny_mode = DENY_WRITE, or if deny_mode = DENY_READ
then it's going to trigger both the if (deny_mode & DENY_READ)
*and* the (deny_mode & DENY_WRITE) conditions.
These problems allowed the original test test_netatalk_lock code to
pass (which was added for BUG: https://bugzilla.samba.org/show_bug.cgi?id=13584
to demonstrate the lock order violation).
This patch refactors the fruit_check_access()
code to be much simpler (IMHO) to understand.
Firstly, pass in the SMB1/2 share mode, not old
DOS deny modes.
Secondly, read all the possible NetAtalk locks
into local variables:
netatalk_already_open_for_reading
netatalk_already_open_with_deny_read
netatalk_already_open_for_writing
netatalk_already_open_with_deny_write
Then do the share mode/access mode checks
with the requested values against any stored
netatalk modes/access modes.
Finally add in NetATalk compatible locks
that represent our share modes/access modes
into the file, with an early return if we don't
have FILE_READ_DATA (in which case we can't
write locks anyway).
The patch is easier to understand by looking
at the completed patched fruit_check_access()
function, rather than trying to look at the
diff.
Signed-off-by: Jeremy Allison <jra@samba.org>
Reviewed-by: Ralph Böhme <slow@samba.org>
2019-02-07 04:49:16 +03:00
if ( ! share_for_read ) {
off = denymode_to_netatalk_brl ( fork_type , DENY_READ ) ;
2019-08-08 20:26:28 +03:00
req_guid . time_hi_and_version = __LINE__ ;
2019-07-01 15:30:15 +03:00
status = do_lock (
2019-06-20 13:20:39 +03:00
fsp ,
2019-08-08 20:26:28 +03:00
talloc_tos ( ) ,
& req_guid ,
2019-06-20 13:20:39 +03:00
fsp - > op - > global - > open_persistent_id ,
1 ,
off ,
READ_LOCK ,
POSIX_LOCK ,
NULL ,
NULL ) ;
2014-06-23 18:59:45 +04:00
s3: VFS: vfs_fruit. Fix the NetAtalk deny mode compatibility code.
This exhibited itself as a problem with OFD locks reported
as:
BUG: https://bugzilla.samba.org/show_bug.cgi?id=13770
However, due to underlying bugs in the vfs_fruit
code the file locks were not being properly applied.
There are two problems in fruit_check_access().
Problem #1:
Inside fruit_check_access() we have:
flags = fcntl(fsp->fh->fd, F_GETFL);
..
if (flags & (O_RDONLY|O_RDWR)) {
We shouldn't be calling fcntl(fsp->fh->fd, ..) directly.
fsp->fh->fd may be a made up number from an underlying
VFS module that has no meaning to a system call.
Secondly, in all POSIX systems - O_RDONLY is defined as
*zero*. O_RDWR = 2.
Which means flags & (O_RDONLY|O_RDWR) becomes (flags & 2),
not what we actually thought.
Problem #2:
deny_mode is *not* a bitmask, it's a set of discrete values.
Inside fruit_check_access() we have:
if (deny_mode & DENY_READ) and also (deny_mode & DENY_WRITE)
However, deny modes are defined as:
/* deny modes */
define DENY_DOS 0
define DENY_ALL 1
define DENY_WRITE 2
define DENY_READ 3
define DENY_NONE 4
define DENY_FCB 7
so if deny_mode = DENY_WRITE, or if deny_mode = DENY_READ
then it's going to trigger both the if (deny_mode & DENY_READ)
*and* the (deny_mode & DENY_WRITE) conditions.
These problems allowed the original test test_netatalk_lock code to
pass (which was added for BUG: https://bugzilla.samba.org/show_bug.cgi?id=13584
to demonstrate the lock order violation).
This patch refactors the fruit_check_access()
code to be much simpler (IMHO) to understand.
Firstly, pass in the SMB1/2 share mode, not old
DOS deny modes.
Secondly, read all the possible NetAtalk locks
into local variables:
netatalk_already_open_for_reading
netatalk_already_open_with_deny_read
netatalk_already_open_for_writing
netatalk_already_open_with_deny_write
Then do the share mode/access mode checks
with the requested values against any stored
netatalk modes/access modes.
Finally add in NetATalk compatible locks
that represent our share modes/access modes
into the file, with an early return if we don't
have FILE_READ_DATA (in which case we can't
write locks anyway).
The patch is easier to understand by looking
at the completed patched fruit_check_access()
function, rather than trying to look at the
diff.
Signed-off-by: Jeremy Allison <jra@samba.org>
Reviewed-by: Ralph Böhme <slow@samba.org>
2019-02-07 04:49:16 +03:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
2014-06-23 18:59:45 +04:00
}
s3: VFS: vfs_fruit. Fix the NetAtalk deny mode compatibility code.
This exhibited itself as a problem with OFD locks reported
as:
BUG: https://bugzilla.samba.org/show_bug.cgi?id=13770
However, due to underlying bugs in the vfs_fruit
code the file locks were not being properly applied.
There are two problems in fruit_check_access().
Problem #1:
Inside fruit_check_access() we have:
flags = fcntl(fsp->fh->fd, F_GETFL);
..
if (flags & (O_RDONLY|O_RDWR)) {
We shouldn't be calling fcntl(fsp->fh->fd, ..) directly.
fsp->fh->fd may be a made up number from an underlying
VFS module that has no meaning to a system call.
Secondly, in all POSIX systems - O_RDONLY is defined as
*zero*. O_RDWR = 2.
Which means flags & (O_RDONLY|O_RDWR) becomes (flags & 2),
not what we actually thought.
Problem #2:
deny_mode is *not* a bitmask, it's a set of discrete values.
Inside fruit_check_access() we have:
if (deny_mode & DENY_READ) and also (deny_mode & DENY_WRITE)
However, deny modes are defined as:
/* deny modes */
define DENY_DOS 0
define DENY_ALL 1
define DENY_WRITE 2
define DENY_READ 3
define DENY_NONE 4
define DENY_FCB 7
so if deny_mode = DENY_WRITE, or if deny_mode = DENY_READ
then it's going to trigger both the if (deny_mode & DENY_READ)
*and* the (deny_mode & DENY_WRITE) conditions.
These problems allowed the original test test_netatalk_lock code to
pass (which was added for BUG: https://bugzilla.samba.org/show_bug.cgi?id=13584
to demonstrate the lock order violation).
This patch refactors the fruit_check_access()
code to be much simpler (IMHO) to understand.
Firstly, pass in the SMB1/2 share mode, not old
DOS deny modes.
Secondly, read all the possible NetAtalk locks
into local variables:
netatalk_already_open_for_reading
netatalk_already_open_with_deny_read
netatalk_already_open_for_writing
netatalk_already_open_with_deny_write
Then do the share mode/access mode checks
with the requested values against any stored
netatalk modes/access modes.
Finally add in NetATalk compatible locks
that represent our share modes/access modes
into the file, with an early return if we don't
have FILE_READ_DATA (in which case we can't
write locks anyway).
The patch is easier to understand by looking
at the completed patched fruit_check_access()
function, rather than trying to look at the
diff.
Signed-off-by: Jeremy Allison <jra@samba.org>
Reviewed-by: Ralph Böhme <slow@samba.org>
2019-02-07 04:49:16 +03:00
}
2014-06-23 18:59:45 +04:00
s3: VFS: vfs_fruit. Fix the NetAtalk deny mode compatibility code.
This exhibited itself as a problem with OFD locks reported
as:
BUG: https://bugzilla.samba.org/show_bug.cgi?id=13770
However, due to underlying bugs in the vfs_fruit
code the file locks were not being properly applied.
There are two problems in fruit_check_access().
Problem #1:
Inside fruit_check_access() we have:
flags = fcntl(fsp->fh->fd, F_GETFL);
..
if (flags & (O_RDONLY|O_RDWR)) {
We shouldn't be calling fcntl(fsp->fh->fd, ..) directly.
fsp->fh->fd may be a made up number from an underlying
VFS module that has no meaning to a system call.
Secondly, in all POSIX systems - O_RDONLY is defined as
*zero*. O_RDWR = 2.
Which means flags & (O_RDONLY|O_RDWR) becomes (flags & 2),
not what we actually thought.
Problem #2:
deny_mode is *not* a bitmask, it's a set of discrete values.
Inside fruit_check_access() we have:
if (deny_mode & DENY_READ) and also (deny_mode & DENY_WRITE)
However, deny modes are defined as:
/* deny modes */
define DENY_DOS 0
define DENY_ALL 1
define DENY_WRITE 2
define DENY_READ 3
define DENY_NONE 4
define DENY_FCB 7
so if deny_mode = DENY_WRITE, or if deny_mode = DENY_READ
then it's going to trigger both the if (deny_mode & DENY_READ)
*and* the (deny_mode & DENY_WRITE) conditions.
These problems allowed the original test test_netatalk_lock code to
pass (which was added for BUG: https://bugzilla.samba.org/show_bug.cgi?id=13584
to demonstrate the lock order violation).
This patch refactors the fruit_check_access()
code to be much simpler (IMHO) to understand.
Firstly, pass in the SMB1/2 share mode, not old
DOS deny modes.
Secondly, read all the possible NetAtalk locks
into local variables:
netatalk_already_open_for_reading
netatalk_already_open_with_deny_read
netatalk_already_open_for_writing
netatalk_already_open_with_deny_write
Then do the share mode/access mode checks
with the requested values against any stored
netatalk modes/access modes.
Finally add in NetATalk compatible locks
that represent our share modes/access modes
into the file, with an early return if we don't
have FILE_READ_DATA (in which case we can't
write locks anyway).
The patch is easier to understand by looking
at the completed patched fruit_check_access()
function, rather than trying to look at the
diff.
Signed-off-by: Jeremy Allison <jra@samba.org>
Reviewed-by: Ralph Böhme <slow@samba.org>
2019-02-07 04:49:16 +03:00
if ( access_mask & FILE_WRITE_DATA ) {
off = access_to_netatalk_brl ( fork_type , FILE_WRITE_DATA ) ;
2019-08-08 20:26:28 +03:00
req_guid . time_hi_and_version = __LINE__ ;
2019-07-01 15:30:15 +03:00
status = do_lock (
2019-06-20 13:20:39 +03:00
fsp ,
2019-08-08 20:26:28 +03:00
talloc_tos ( ) ,
& req_guid ,
2019-06-20 13:20:39 +03:00
fsp - > op - > global - > open_persistent_id ,
1 ,
off ,
READ_LOCK ,
POSIX_LOCK ,
NULL ,
NULL ) ;
2014-06-23 18:59:45 +04:00
s3: VFS: vfs_fruit. Fix the NetAtalk deny mode compatibility code.
This exhibited itself as a problem with OFD locks reported
as:
BUG: https://bugzilla.samba.org/show_bug.cgi?id=13770
However, due to underlying bugs in the vfs_fruit
code the file locks were not being properly applied.
There are two problems in fruit_check_access().
Problem #1:
Inside fruit_check_access() we have:
flags = fcntl(fsp->fh->fd, F_GETFL);
..
if (flags & (O_RDONLY|O_RDWR)) {
We shouldn't be calling fcntl(fsp->fh->fd, ..) directly.
fsp->fh->fd may be a made up number from an underlying
VFS module that has no meaning to a system call.
Secondly, in all POSIX systems - O_RDONLY is defined as
*zero*. O_RDWR = 2.
Which means flags & (O_RDONLY|O_RDWR) becomes (flags & 2),
not what we actually thought.
Problem #2:
deny_mode is *not* a bitmask, it's a set of discrete values.
Inside fruit_check_access() we have:
if (deny_mode & DENY_READ) and also (deny_mode & DENY_WRITE)
However, deny modes are defined as:
/* deny modes */
define DENY_DOS 0
define DENY_ALL 1
define DENY_WRITE 2
define DENY_READ 3
define DENY_NONE 4
define DENY_FCB 7
so if deny_mode = DENY_WRITE, or if deny_mode = DENY_READ
then it's going to trigger both the if (deny_mode & DENY_READ)
*and* the (deny_mode & DENY_WRITE) conditions.
These problems allowed the original test test_netatalk_lock code to
pass (which was added for BUG: https://bugzilla.samba.org/show_bug.cgi?id=13584
to demonstrate the lock order violation).
This patch refactors the fruit_check_access()
code to be much simpler (IMHO) to understand.
Firstly, pass in the SMB1/2 share mode, not old
DOS deny modes.
Secondly, read all the possible NetAtalk locks
into local variables:
netatalk_already_open_for_reading
netatalk_already_open_with_deny_read
netatalk_already_open_for_writing
netatalk_already_open_with_deny_write
Then do the share mode/access mode checks
with the requested values against any stored
netatalk modes/access modes.
Finally add in NetATalk compatible locks
that represent our share modes/access modes
into the file, with an early return if we don't
have FILE_READ_DATA (in which case we can't
write locks anyway).
The patch is easier to understand by looking
at the completed patched fruit_check_access()
function, rather than trying to look at the
diff.
Signed-off-by: Jeremy Allison <jra@samba.org>
Reviewed-by: Ralph Böhme <slow@samba.org>
2019-02-07 04:49:16 +03:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
2014-06-23 18:59:45 +04:00
}
s3: VFS: vfs_fruit. Fix the NetAtalk deny mode compatibility code.
This exhibited itself as a problem with OFD locks reported
as:
BUG: https://bugzilla.samba.org/show_bug.cgi?id=13770
However, due to underlying bugs in the vfs_fruit
code the file locks were not being properly applied.
There are two problems in fruit_check_access().
Problem #1:
Inside fruit_check_access() we have:
flags = fcntl(fsp->fh->fd, F_GETFL);
..
if (flags & (O_RDONLY|O_RDWR)) {
We shouldn't be calling fcntl(fsp->fh->fd, ..) directly.
fsp->fh->fd may be a made up number from an underlying
VFS module that has no meaning to a system call.
Secondly, in all POSIX systems - O_RDONLY is defined as
*zero*. O_RDWR = 2.
Which means flags & (O_RDONLY|O_RDWR) becomes (flags & 2),
not what we actually thought.
Problem #2:
deny_mode is *not* a bitmask, it's a set of discrete values.
Inside fruit_check_access() we have:
if (deny_mode & DENY_READ) and also (deny_mode & DENY_WRITE)
However, deny modes are defined as:
/* deny modes */
define DENY_DOS 0
define DENY_ALL 1
define DENY_WRITE 2
define DENY_READ 3
define DENY_NONE 4
define DENY_FCB 7
so if deny_mode = DENY_WRITE, or if deny_mode = DENY_READ
then it's going to trigger both the if (deny_mode & DENY_READ)
*and* the (deny_mode & DENY_WRITE) conditions.
These problems allowed the original test test_netatalk_lock code to
pass (which was added for BUG: https://bugzilla.samba.org/show_bug.cgi?id=13584
to demonstrate the lock order violation).
This patch refactors the fruit_check_access()
code to be much simpler (IMHO) to understand.
Firstly, pass in the SMB1/2 share mode, not old
DOS deny modes.
Secondly, read all the possible NetAtalk locks
into local variables:
netatalk_already_open_for_reading
netatalk_already_open_with_deny_read
netatalk_already_open_for_writing
netatalk_already_open_with_deny_write
Then do the share mode/access mode checks
with the requested values against any stored
netatalk modes/access modes.
Finally add in NetATalk compatible locks
that represent our share modes/access modes
into the file, with an early return if we don't
have FILE_READ_DATA (in which case we can't
write locks anyway).
The patch is easier to understand by looking
at the completed patched fruit_check_access()
function, rather than trying to look at the
diff.
Signed-off-by: Jeremy Allison <jra@samba.org>
Reviewed-by: Ralph Böhme <slow@samba.org>
2019-02-07 04:49:16 +03:00
}
2018-08-06 15:33:34 +03:00
s3: VFS: vfs_fruit. Fix the NetAtalk deny mode compatibility code.
This exhibited itself as a problem with OFD locks reported
as:
BUG: https://bugzilla.samba.org/show_bug.cgi?id=13770
However, due to underlying bugs in the vfs_fruit
code the file locks were not being properly applied.
There are two problems in fruit_check_access().
Problem #1:
Inside fruit_check_access() we have:
flags = fcntl(fsp->fh->fd, F_GETFL);
..
if (flags & (O_RDONLY|O_RDWR)) {
We shouldn't be calling fcntl(fsp->fh->fd, ..) directly.
fsp->fh->fd may be a made up number from an underlying
VFS module that has no meaning to a system call.
Secondly, in all POSIX systems - O_RDONLY is defined as
*zero*. O_RDWR = 2.
Which means flags & (O_RDONLY|O_RDWR) becomes (flags & 2),
not what we actually thought.
Problem #2:
deny_mode is *not* a bitmask, it's a set of discrete values.
Inside fruit_check_access() we have:
if (deny_mode & DENY_READ) and also (deny_mode & DENY_WRITE)
However, deny modes are defined as:
/* deny modes */
define DENY_DOS 0
define DENY_ALL 1
define DENY_WRITE 2
define DENY_READ 3
define DENY_NONE 4
define DENY_FCB 7
so if deny_mode = DENY_WRITE, or if deny_mode = DENY_READ
then it's going to trigger both the if (deny_mode & DENY_READ)
*and* the (deny_mode & DENY_WRITE) conditions.
These problems allowed the original test test_netatalk_lock code to
pass (which was added for BUG: https://bugzilla.samba.org/show_bug.cgi?id=13584
to demonstrate the lock order violation).
This patch refactors the fruit_check_access()
code to be much simpler (IMHO) to understand.
Firstly, pass in the SMB1/2 share mode, not old
DOS deny modes.
Secondly, read all the possible NetAtalk locks
into local variables:
netatalk_already_open_for_reading
netatalk_already_open_with_deny_read
netatalk_already_open_for_writing
netatalk_already_open_with_deny_write
Then do the share mode/access mode checks
with the requested values against any stored
netatalk modes/access modes.
Finally add in NetATalk compatible locks
that represent our share modes/access modes
into the file, with an early return if we don't
have FILE_READ_DATA (in which case we can't
write locks anyway).
The patch is easier to understand by looking
at the completed patched fruit_check_access()
function, rather than trying to look at the
diff.
Signed-off-by: Jeremy Allison <jra@samba.org>
Reviewed-by: Ralph Böhme <slow@samba.org>
2019-02-07 04:49:16 +03:00
if ( ! share_for_write ) {
off = denymode_to_netatalk_brl ( fork_type , DENY_WRITE ) ;
2019-08-08 20:26:28 +03:00
req_guid . time_hi_and_version = __LINE__ ;
2019-07-01 15:30:15 +03:00
status = do_lock (
2019-06-20 13:20:39 +03:00
fsp ,
2019-08-08 20:26:28 +03:00
talloc_tos ( ) ,
& req_guid ,
2019-06-20 13:20:39 +03:00
fsp - > op - > global - > open_persistent_id ,
1 ,
off ,
READ_LOCK ,
POSIX_LOCK ,
NULL ,
NULL ) ;
2018-08-06 15:33:34 +03:00
s3: VFS: vfs_fruit. Fix the NetAtalk deny mode compatibility code.
This exhibited itself as a problem with OFD locks reported
as:
BUG: https://bugzilla.samba.org/show_bug.cgi?id=13770
However, due to underlying bugs in the vfs_fruit
code the file locks were not being properly applied.
There are two problems in fruit_check_access().
Problem #1:
Inside fruit_check_access() we have:
flags = fcntl(fsp->fh->fd, F_GETFL);
..
if (flags & (O_RDONLY|O_RDWR)) {
We shouldn't be calling fcntl(fsp->fh->fd, ..) directly.
fsp->fh->fd may be a made up number from an underlying
VFS module that has no meaning to a system call.
Secondly, in all POSIX systems - O_RDONLY is defined as
*zero*. O_RDWR = 2.
Which means flags & (O_RDONLY|O_RDWR) becomes (flags & 2),
not what we actually thought.
Problem #2:
deny_mode is *not* a bitmask, it's a set of discrete values.
Inside fruit_check_access() we have:
if (deny_mode & DENY_READ) and also (deny_mode & DENY_WRITE)
However, deny modes are defined as:
/* deny modes */
define DENY_DOS 0
define DENY_ALL 1
define DENY_WRITE 2
define DENY_READ 3
define DENY_NONE 4
define DENY_FCB 7
so if deny_mode = DENY_WRITE, or if deny_mode = DENY_READ
then it's going to trigger both the if (deny_mode & DENY_READ)
*and* the (deny_mode & DENY_WRITE) conditions.
These problems allowed the original test test_netatalk_lock code to
pass (which was added for BUG: https://bugzilla.samba.org/show_bug.cgi?id=13584
to demonstrate the lock order violation).
This patch refactors the fruit_check_access()
code to be much simpler (IMHO) to understand.
Firstly, pass in the SMB1/2 share mode, not old
DOS deny modes.
Secondly, read all the possible NetAtalk locks
into local variables:
netatalk_already_open_for_reading
netatalk_already_open_with_deny_read
netatalk_already_open_for_writing
netatalk_already_open_with_deny_write
Then do the share mode/access mode checks
with the requested values against any stored
netatalk modes/access modes.
Finally add in NetATalk compatible locks
that represent our share modes/access modes
into the file, with an early return if we don't
have FILE_READ_DATA (in which case we can't
write locks anyway).
The patch is easier to understand by looking
at the completed patched fruit_check_access()
function, rather than trying to look at the
diff.
Signed-off-by: Jeremy Allison <jra@samba.org>
Reviewed-by: Ralph Böhme <slow@samba.org>
2019-02-07 04:49:16 +03:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
2014-06-23 18:59:45 +04:00
}
}
s3: VFS: vfs_fruit. Fix the NetAtalk deny mode compatibility code.
This exhibited itself as a problem with OFD locks reported
as:
BUG: https://bugzilla.samba.org/show_bug.cgi?id=13770
However, due to underlying bugs in the vfs_fruit
code the file locks were not being properly applied.
There are two problems in fruit_check_access().
Problem #1:
Inside fruit_check_access() we have:
flags = fcntl(fsp->fh->fd, F_GETFL);
..
if (flags & (O_RDONLY|O_RDWR)) {
We shouldn't be calling fcntl(fsp->fh->fd, ..) directly.
fsp->fh->fd may be a made up number from an underlying
VFS module that has no meaning to a system call.
Secondly, in all POSIX systems - O_RDONLY is defined as
*zero*. O_RDWR = 2.
Which means flags & (O_RDONLY|O_RDWR) becomes (flags & 2),
not what we actually thought.
Problem #2:
deny_mode is *not* a bitmask, it's a set of discrete values.
Inside fruit_check_access() we have:
if (deny_mode & DENY_READ) and also (deny_mode & DENY_WRITE)
However, deny modes are defined as:
/* deny modes */
define DENY_DOS 0
define DENY_ALL 1
define DENY_WRITE 2
define DENY_READ 3
define DENY_NONE 4
define DENY_FCB 7
so if deny_mode = DENY_WRITE, or if deny_mode = DENY_READ
then it's going to trigger both the if (deny_mode & DENY_READ)
*and* the (deny_mode & DENY_WRITE) conditions.
These problems allowed the original test test_netatalk_lock code to
pass (which was added for BUG: https://bugzilla.samba.org/show_bug.cgi?id=13584
to demonstrate the lock order violation).
This patch refactors the fruit_check_access()
code to be much simpler (IMHO) to understand.
Firstly, pass in the SMB1/2 share mode, not old
DOS deny modes.
Secondly, read all the possible NetAtalk locks
into local variables:
netatalk_already_open_for_reading
netatalk_already_open_with_deny_read
netatalk_already_open_for_writing
netatalk_already_open_with_deny_write
Then do the share mode/access mode checks
with the requested values against any stored
netatalk modes/access modes.
Finally add in NetATalk compatible locks
that represent our share modes/access modes
into the file, with an early return if we don't
have FILE_READ_DATA (in which case we can't
write locks anyway).
The patch is easier to understand by looking
at the completed patched fruit_check_access()
function, rather than trying to look at the
diff.
Signed-off-by: Jeremy Allison <jra@samba.org>
Reviewed-by: Ralph Böhme <slow@samba.org>
2019-02-07 04:49:16 +03:00
return NT_STATUS_OK ;
2014-06-23 18:59:45 +04:00
}
2014-11-26 20:11:17 +03:00
static NTSTATUS check_aapl ( vfs_handle_struct * handle ,
struct smb_request * req ,
const struct smb2_create_blobs * in_context_blobs ,
struct smb2_create_blobs * out_context_blobs )
{
struct fruit_config_data * config ;
NTSTATUS status ;
struct smb2_create_blob * aapl = NULL ;
uint32_t cmd ;
bool ok ;
uint8_t p [ 16 ] ;
DATA_BLOB blob = data_blob_talloc ( req , NULL , 0 ) ;
uint64_t req_bitmap , client_caps ;
uint64_t server_caps = SMB2_CRTCTX_AAPL_UNIX_BASED ;
smb_ucs2_t * model ;
size_t modellen ;
SMB_VFS_HANDLE_GET_DATA ( handle , config , struct fruit_config_data ,
return NT_STATUS_UNSUCCESSFUL ) ;
if ( ! config - > use_aapl
| | in_context_blobs = = NULL
| | out_context_blobs = = NULL ) {
return NT_STATUS_OK ;
}
aapl = smb2_create_blob_find ( in_context_blobs ,
SMB2_CREATE_TAG_AAPL ) ;
if ( aapl = = NULL ) {
return NT_STATUS_OK ;
}
if ( aapl - > data . length ! = 24 ) {
2015-11-08 12:16:04 +03:00
DEBUG ( 1 , ( " unexpected AAPL ctxt length: %ju \n " ,
2014-11-26 20:11:17 +03:00
( uintmax_t ) aapl - > data . length ) ) ;
return NT_STATUS_INVALID_PARAMETER ;
}
cmd = IVAL ( aapl - > data . data , 0 ) ;
if ( cmd ! = SMB2_CRTCTX_AAPL_SERVER_QUERY ) {
DEBUG ( 1 , ( " unsupported AAPL cmd: %d \n " , cmd ) ) ;
return NT_STATUS_INVALID_PARAMETER ;
}
req_bitmap = BVAL ( aapl - > data . data , 8 ) ;
client_caps = BVAL ( aapl - > data . data , 16 ) ;
SIVAL ( p , 0 , SMB2_CRTCTX_AAPL_SERVER_QUERY ) ;
SIVAL ( p , 4 , 0 ) ;
SBVAL ( p , 8 , req_bitmap ) ;
ok = data_blob_append ( req , & blob , p , 16 ) ;
if ( ! ok ) {
return NT_STATUS_UNSUCCESSFUL ;
}
if ( req_bitmap & SMB2_CRTCTX_AAPL_SERVER_CAPS ) {
if ( ( client_caps & SMB2_CRTCTX_AAPL_SUPPORTS_READ_DIR_ATTR ) & &
2023-04-05 17:59:28 +03:00
( handle - > conn - > fs_capabilities & FILE_NAMED_STREAMS ) ) {
2014-11-26 20:11:17 +03:00
server_caps | = SMB2_CRTCTX_AAPL_SUPPORTS_READ_DIR_ATTR ;
config - > readdir_attr_enabled = true ;
}
2015-04-22 23:29:16 +03:00
if ( config - > use_copyfile ) {
server_caps | = SMB2_CRTCTX_AAPL_SUPPORTS_OSX_COPYFILE ;
config - > copyfile_enabled = true ;
}
2014-11-26 20:11:17 +03:00
/*
* The client doesn ' t set the flag , so we can ' t check
* for it and just set it unconditionally
*/
2015-03-25 17:09:02 +03:00
if ( config - > unix_info_enabled ) {
server_caps | = SMB2_CRTCTX_AAPL_SUPPORTS_NFS_ACE ;
}
2014-11-26 20:11:17 +03:00
SBVAL ( p , 0 , server_caps ) ;
ok = data_blob_append ( req , & blob , p , 8 ) ;
if ( ! ok ) {
return NT_STATUS_UNSUCCESSFUL ;
}
}
if ( req_bitmap & SMB2_CRTCTX_AAPL_VOLUME_CAPS ) {
2023-04-05 17:59:28 +03:00
int val = lp_case_sensitive ( SNUM ( handle - > conn ) ) ;
2017-04-19 14:12:55 +03:00
uint64_t caps = 0 ;
switch ( val ) {
case Auto :
break ;
case True :
caps | = SMB2_CRTCTX_AAPL_CASE_SENSITIVE ;
break ;
default :
break ;
}
2016-11-14 21:14:44 +03:00
if ( config - > time_machine ) {
caps | = SMB2_CRTCTX_AAPL_FULL_SYNC ;
}
2017-04-19 14:12:55 +03:00
SBVAL ( p , 0 , caps ) ;
2014-11-26 20:11:17 +03:00
ok = data_blob_append ( req , & blob , p , 8 ) ;
if ( ! ok ) {
return NT_STATUS_UNSUCCESSFUL ;
}
}
if ( req_bitmap & SMB2_CRTCTX_AAPL_MODEL_INFO ) {
ok = convert_string_talloc ( req ,
CH_UNIX , CH_UTF16LE ,
2017-06-28 19:10:28 +03:00
config - > model , strlen ( config - > model ) ,
2014-11-26 20:11:17 +03:00
& model , & modellen ) ;
if ( ! ok ) {
return NT_STATUS_UNSUCCESSFUL ;
}
SIVAL ( p , 0 , 0 ) ;
SIVAL ( p + 4 , 0 , modellen ) ;
ok = data_blob_append ( req , & blob , p , 8 ) ;
if ( ! ok ) {
talloc_free ( model ) ;
return NT_STATUS_UNSUCCESSFUL ;
}
ok = data_blob_append ( req , & blob , model , modellen ) ;
talloc_free ( model ) ;
if ( ! ok ) {
return NT_STATUS_UNSUCCESSFUL ;
}
}
status = smb2_create_blob_add ( out_context_blobs ,
out_context_blobs ,
SMB2_CREATE_TAG_AAPL ,
blob ) ;
2015-11-25 11:12:55 +03:00
if ( NT_STATUS_IS_OK ( status ) ) {
2017-02-28 11:39:37 +03:00
global_fruit_config . nego_aapl = true ;
2015-11-25 11:12:55 +03:00
}
2014-11-26 20:11:17 +03:00
return status ;
}
2016-12-01 19:04:35 +03:00
static bool readdir_attr_meta_finderi_stream (
struct vfs_handle_struct * handle ,
const struct smb_filename * smb_fname ,
AfpInfo * ai )
{
2016-12-09 19:24:18 +03:00
struct smb_filename * stream_name = NULL ;
files_struct * fsp = NULL ;
ssize_t nread ;
NTSTATUS status ;
bool ok ;
uint8_t buf [ AFP_INFO_SIZE ] ;
2021-01-25 18:10:12 +03:00
status = synthetic_pathref ( talloc_tos ( ) ,
handle - > conn - > cwd_fsp ,
smb_fname - > base_name ,
AFPINFO_STREAM_NAME ,
NULL ,
smb_fname - > twrp ,
smb_fname - > flags ,
& stream_name ) ;
2020-11-12 14:37:52 +03:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
return false ;
}
2016-12-09 19:24:18 +03:00
status = SMB_VFS_CREATE_FILE (
handle - > conn , /* conn */
NULL , /* req */
2021-11-23 14:29:17 +03:00
NULL , /* dirfsp */
2016-12-09 19:24:18 +03:00
stream_name , /* fname */
FILE_READ_DATA , /* access_mask */
( FILE_SHARE_READ | FILE_SHARE_WRITE | /* share_access */
FILE_SHARE_DELETE ) ,
FILE_OPEN , /* create_disposition*/
0 , /* create_options */
0 , /* file_attributes */
INTERNAL_OPEN_ONLY , /* oplock_request */
NULL , /* lease */
0 , /* allocation_size */
0 , /* private_flags */
NULL , /* sd */
NULL , /* ea_list */
& fsp , /* result */
NULL , /* pinfo */
NULL , NULL ) ; /* create context */
TALLOC_FREE ( stream_name ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return false ;
}
nread = SMB_VFS_PREAD ( fsp , & buf [ 0 ] , AFP_INFO_SIZE , 0 ) ;
if ( nread ! = AFP_INFO_SIZE ) {
DBG_ERR ( " short read [%s] [%zd/%d] \n " ,
smb_fname_str_dbg ( stream_name ) , nread , AFP_INFO_SIZE ) ;
ok = false ;
goto fail ;
}
memcpy ( & ai - > afpi_FinderInfo [ 0 ] , & buf [ AFP_OFF_FinderInfo ] ,
AFP_FinderSize ) ;
ok = true ;
fail :
if ( fsp ! = NULL ) {
2022-02-01 19:47:29 +03:00
close_file_free ( NULL , & fsp , NORMAL_CLOSE ) ;
2016-12-09 19:24:18 +03:00
}
return ok ;
2016-12-01 19:04:35 +03:00
}
static bool readdir_attr_meta_finderi_netatalk (
struct vfs_handle_struct * handle ,
const struct smb_filename * smb_fname ,
AfpInfo * ai )
{
struct adouble * ad = NULL ;
char * p = NULL ;
2021-07-02 23:52:02 +03:00
ad = ad_get_meta_fsp ( talloc_tos ( ) , handle , smb_fname ) ;
2016-12-01 19:04:35 +03:00
if ( ad = = NULL ) {
return false ;
}
p = ad_get_entry ( ad , ADEID_FINDERI ) ;
if ( p = = NULL ) {
DBG_ERR ( " No ADEID_FINDERI for [%s] \n " , smb_fname - > base_name ) ;
TALLOC_FREE ( ad ) ;
return false ;
}
memcpy ( & ai - > afpi_FinderInfo [ 0 ] , p , AFP_FinderSize ) ;
TALLOC_FREE ( ad ) ;
return true ;
}
static bool readdir_attr_meta_finderi ( struct vfs_handle_struct * handle ,
const struct smb_filename * smb_fname ,
struct readdir_attr_data * attr_data )
{
struct fruit_config_data * config = NULL ;
uint32_t date_added ;
AfpInfo ai = { 0 } ;
bool ok ;
SMB_VFS_HANDLE_GET_DATA ( handle , config ,
struct fruit_config_data ,
return false ) ;
switch ( config - > meta ) {
case FRUIT_META_NETATALK :
ok = readdir_attr_meta_finderi_netatalk (
handle , smb_fname , & ai ) ;
break ;
case FRUIT_META_STREAM :
ok = readdir_attr_meta_finderi_stream (
handle , smb_fname , & ai ) ;
break ;
default :
DBG_ERR ( " Unexpected meta config [%d] \n " , config - > meta ) ;
return false ;
}
if ( ! ok ) {
/* Don't bother with errors, it's likely ENOENT */
return true ;
}
if ( S_ISREG ( smb_fname - > st . st_ex_mode ) ) {
/* finder_type */
memcpy ( & attr_data - > attr_data . aapl . finder_info [ 0 ] ,
& ai . afpi_FinderInfo [ 0 ] , 4 ) ;
/* finder_creator */
memcpy ( & attr_data - > attr_data . aapl . finder_info [ 0 ] + 4 ,
& ai . afpi_FinderInfo [ 4 ] , 4 ) ;
}
/* finder_flags */
memcpy ( & attr_data - > attr_data . aapl . finder_info [ 0 ] + 8 ,
& ai . afpi_FinderInfo [ 8 ] , 2 ) ;
/* finder_ext_flags */
memcpy ( & attr_data - > attr_data . aapl . finder_info [ 0 ] + 10 ,
& ai . afpi_FinderInfo [ 24 ] , 2 ) ;
/* creation date */
date_added = convert_time_t_to_uint32_t (
smb_fname - > st . st_ex_btime . tv_sec - AD_DATE_DELTA ) ;
RSIVAL ( & attr_data - > attr_data . aapl . finder_info [ 0 ] , 12 , date_added ) ;
return true ;
}
2016-12-02 19:00:03 +03:00
static uint64_t readdir_attr_rfork_size_adouble (
struct vfs_handle_struct * handle ,
const struct smb_filename * smb_fname )
{
struct adouble * ad = NULL ;
uint64_t rfork_size ;
2017-05-25 21:38:26 +03:00
ad = ad_get ( talloc_tos ( ) , handle , smb_fname ,
2016-12-02 19:00:03 +03:00
ADOUBLE_RSRC ) ;
if ( ad = = NULL ) {
return 0 ;
}
rfork_size = ad_getentrylen ( ad , ADEID_RFORK ) ;
TALLOC_FREE ( ad ) ;
return rfork_size ;
}
static uint64_t readdir_attr_rfork_size_stream (
struct vfs_handle_struct * handle ,
const struct smb_filename * smb_fname )
{
struct smb_filename * stream_name = NULL ;
int ret ;
uint64_t rfork_size ;
stream_name = synthetic_smb_fname ( talloc_tos ( ) ,
smb_fname - > base_name ,
AFPRESOURCE_STREAM_NAME ,
2020-05-03 16:03:22 +03:00
NULL ,
2020-04-30 12:48:32 +03:00
smb_fname - > twrp ,
2020-05-03 16:03:22 +03:00
0 ) ;
2016-12-02 19:00:03 +03:00
if ( stream_name = = NULL ) {
return 0 ;
}
ret = SMB_VFS_STAT ( handle - > conn , stream_name ) ;
if ( ret ! = 0 ) {
TALLOC_FREE ( stream_name ) ;
return 0 ;
}
rfork_size = stream_name - > st . st_ex_size ;
TALLOC_FREE ( stream_name ) ;
return rfork_size ;
}
static uint64_t readdir_attr_rfork_size ( struct vfs_handle_struct * handle ,
const struct smb_filename * smb_fname )
{
struct fruit_config_data * config = NULL ;
uint64_t rfork_size ;
SMB_VFS_HANDLE_GET_DATA ( handle , config ,
struct fruit_config_data ,
return 0 ) ;
switch ( config - > rsrc ) {
case FRUIT_RSRC_ADFILE :
rfork_size = readdir_attr_rfork_size_adouble ( handle ,
smb_fname ) ;
break ;
2019-05-22 19:08:14 +03:00
case FRUIT_RSRC_XATTR :
2019-05-22 18:02:20 +03:00
case FRUIT_RSRC_STREAM :
2016-12-02 19:00:03 +03:00
rfork_size = readdir_attr_rfork_size_stream ( handle ,
smb_fname ) ;
break ;
default :
DBG_ERR ( " Unexpected rsrc config [%d] \n " , config - > rsrc ) ;
rfork_size = 0 ;
break ;
}
return rfork_size ;
}
2014-11-26 20:11:17 +03:00
static NTSTATUS readdir_attr_macmeta ( struct vfs_handle_struct * handle ,
const struct smb_filename * smb_fname ,
struct readdir_attr_data * attr_data )
{
NTSTATUS status = NT_STATUS_OK ;
struct fruit_config_data * config = NULL ;
2016-12-01 19:04:35 +03:00
bool ok ;
2014-11-26 20:11:17 +03:00
SMB_VFS_HANDLE_GET_DATA ( handle , config ,
struct fruit_config_data ,
return NT_STATUS_UNSUCCESSFUL ) ;
/* Ensure we return a default value in the creation_date field */
RSIVAL ( & attr_data - > attr_data . aapl . finder_info , 12 , AD_DATE_START ) ;
/*
* Resource fork length
*/
if ( config - > readdir_attr_rsize ) {
2016-12-02 19:00:03 +03:00
uint64_t rfork_size ;
rfork_size = readdir_attr_rfork_size ( handle , smb_fname ) ;
attr_data - > attr_data . aapl . rfork_size = rfork_size ;
2014-11-26 20:11:17 +03:00
}
/*
* FinderInfo
*/
if ( config - > readdir_attr_finder_info ) {
2016-12-01 19:04:35 +03:00
ok = readdir_attr_meta_finderi ( handle , smb_fname , attr_data ) ;
if ( ! ok ) {
status = NT_STATUS_INTERNAL_ERROR ;
2014-11-26 20:11:17 +03:00
}
}
return status ;
}
2018-03-15 19:52:30 +03:00
static NTSTATUS remove_virtual_nfs_aces ( struct security_descriptor * psd )
{
NTSTATUS status ;
uint32_t i ;
if ( psd - > dacl = = NULL ) {
return NT_STATUS_OK ;
}
for ( i = 0 ; i < psd - > dacl - > num_aces ; i + + ) {
/* MS NFS style mode/uid/gid */
2018-03-20 01:46:41 +03:00
int cmp = dom_sid_compare_domain (
2018-03-15 19:52:30 +03:00
& global_sid_Unix_NFS ,
2018-03-20 01:46:41 +03:00
& psd - > dacl - > aces [ i ] . trustee ) ;
if ( cmp ! = 0 ) {
2018-03-15 19:52:30 +03:00
/* Normal ACE entry. */
continue ;
}
/*
* security_descriptor_dacl_del ( )
* * must * return NT_STATUS_OK as we know
* we have something to remove .
*/
status = security_descriptor_dacl_del ( psd ,
& psd - > dacl - > aces [ i ] . trustee ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DBG_WARNING ( " failed to remove MS NFS style ACE: %s \n " ,
nt_errstr ( status ) ) ;
return status ;
}
/*
* security_descriptor_dacl_del ( ) may delete more
* then one entry subsequent to this one if the
* SID matches , but we only need to ensure that
* we stay looking at the same element in the array .
*/
i - - ;
}
return NT_STATUS_OK ;
}
2014-11-26 20:11:17 +03:00
/* Search MS NFS style ACE with UNIX mode */
static NTSTATUS check_ms_nfs ( vfs_handle_struct * handle ,
files_struct * fsp ,
2018-03-03 00:53:55 +03:00
struct security_descriptor * psd ,
2014-11-26 20:11:17 +03:00
mode_t * pmode ,
bool * pdo_chmod )
{
2016-05-19 19:42:04 +03:00
uint32_t i ;
2014-11-26 20:11:17 +03:00
struct fruit_config_data * config = NULL ;
* pdo_chmod = false ;
SMB_VFS_HANDLE_GET_DATA ( handle , config ,
struct fruit_config_data ,
return NT_STATUS_UNSUCCESSFUL ) ;
2017-07-12 10:33:59 +03:00
if ( ! global_fruit_config . nego_aapl ) {
return NT_STATUS_OK ;
}
2014-11-26 20:11:17 +03:00
if ( psd - > dacl = = NULL | | ! config - > unix_info_enabled ) {
return NT_STATUS_OK ;
}
for ( i = 0 ; i < psd - > dacl - > num_aces ; i + + ) {
if ( dom_sid_compare_domain (
& global_sid_Unix_NFS_Mode ,
& psd - > dacl - > aces [ i ] . trustee ) = = 0 ) {
* pmode = ( mode_t ) psd - > dacl - > aces [ i ] . trustee . sub_auths [ 2 ] ;
* pmode & = ( S_IRWXU | S_IRWXG | S_IRWXO ) ;
* pdo_chmod = true ;
DEBUG ( 10 , ( " MS NFS chmod request %s, %04o \n " ,
2015-05-06 17:25:51 +03:00
fsp_str_dbg ( fsp ) , ( unsigned ) ( * pmode ) ) ) ;
2014-11-26 20:11:17 +03:00
break ;
}
}
2018-03-03 00:53:55 +03:00
/*
* Remove any incoming virtual ACE entries generated by
* fruit_fget_nt_acl ( ) .
*/
2018-03-15 19:54:41 +03:00
return remove_virtual_nfs_aces ( psd ) ;
2014-11-26 20:11:17 +03:00
}
2014-06-23 18:59:45 +04:00
/****************************************************************************
* VFS ops
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static int fruit_connect ( vfs_handle_struct * handle ,
const char * service ,
const char * user )
{
int rc ;
char * list = NULL , * newlist = NULL ;
struct fruit_config_data * config ;
2019-11-04 14:29:44 +03:00
const struct loadparm_substitution * lp_sub =
loadparm_s3_global_substitution ( ) ;
2014-06-23 18:59:45 +04:00
DEBUG ( 10 , ( " fruit_connect \n " ) ) ;
rc = SMB_VFS_NEXT_CONNECT ( handle , service , user ) ;
if ( rc < 0 ) {
return rc ;
}
rc = init_fruit_config ( handle ) ;
if ( rc ! = 0 ) {
return rc ;
}
SMB_VFS_HANDLE_GET_DATA ( handle , config ,
struct fruit_config_data , return - 1 ) ;
2015-05-09 09:31:24 +03:00
if ( config - > veto_appledouble ) {
2019-11-04 14:29:44 +03:00
list = lp_veto_files ( talloc_tos ( ) , lp_sub , SNUM ( handle - > conn ) ) ;
2015-05-09 09:31:24 +03:00
if ( list ) {
if ( strstr ( list , " / " ADOUBLE_NAME_PREFIX " */ " ) = = NULL ) {
newlist = talloc_asprintf (
list ,
" %s/ " ADOUBLE_NAME_PREFIX " */ " ,
list ) ;
lp_do_parameter ( SNUM ( handle - > conn ) ,
" veto files " ,
newlist ) ;
}
} else {
lp_do_parameter ( SNUM ( handle - > conn ) ,
" veto files " ,
" / " ADOUBLE_NAME_PREFIX " */ " ) ;
}
TALLOC_FREE ( list ) ;
}
2014-06-23 18:59:45 +04:00
if ( config - > encoding = = FRUIT_ENC_NATIVE ) {
2017-10-10 20:13:36 +03:00
lp_do_parameter ( SNUM ( handle - > conn ) ,
" catia:mappings " ,
2019-07-09 20:26:01 +03:00
macos_string_replace_map ) ;
2014-06-23 18:59:45 +04:00
}
2016-11-14 21:14:44 +03:00
if ( config - > time_machine ) {
DBG_NOTICE ( " Enabling durable handles for Time Machine "
" support on [%s] \n " , service ) ;
lp_do_parameter ( SNUM ( handle - > conn ) , " durable handles " , " yes " ) ;
lp_do_parameter ( SNUM ( handle - > conn ) , " kernel oplocks " , " no " ) ;
lp_do_parameter ( SNUM ( handle - > conn ) , " kernel share modes " , " no " ) ;
if ( ! lp_strict_sync ( SNUM ( handle - > conn ) ) ) {
DBG_WARNING ( " Time Machine without strict sync is not "
" recommended! \n " ) ;
}
lp_do_parameter ( SNUM ( handle - > conn ) , " posix locking " , " no " ) ;
}
2014-06-23 18:59:45 +04:00
return rc ;
}
2020-12-08 18:29:10 +03:00
static void fio_ref_destroy_fn ( void * p_data )
{
struct fio * ref_fio = ( struct fio * ) p_data ;
if ( ref_fio - > real_fio ! = NULL ) {
SMB_ASSERT ( ref_fio - > real_fio - > ad_fsp = = ref_fio - > fsp ) ;
ref_fio - > real_fio - > ad_fsp = NULL ;
ref_fio - > real_fio = NULL ;
}
}
static void fio_close_ad_fsp ( struct fio * fio )
{
if ( fio - > ad_fsp ! = NULL ) {
fd_close ( fio - > ad_fsp ) ;
file_free ( NULL , fio - > ad_fsp ) ;
/* fio_ref_destroy_fn() should have cleared this */
SMB_ASSERT ( fio - > ad_fsp = = NULL ) ;
}
}
static void fio_destroy_fn ( void * p_data )
{
struct fio * fio = ( struct fio * ) p_data ;
fio_close_ad_fsp ( fio ) ;
}
2016-11-29 18:56:00 +03:00
static int fruit_open_meta_stream ( vfs_handle_struct * handle ,
2020-05-20 18:51:23 +03:00
const struct files_struct * dirfsp ,
2020-05-21 21:56:14 +03:00
const struct smb_filename * smb_fname ,
2016-11-29 18:56:00 +03:00
files_struct * fsp ,
int flags ,
mode_t mode )
{
2018-08-22 16:25:26 +03:00
struct fruit_config_data * config = NULL ;
struct fio * fio = NULL ;
2022-06-03 16:53:29 +03:00
struct vfs_open_how how = {
. flags = flags & ~ O_CREAT ,
. mode = mode ,
} ;
2018-08-22 16:25:26 +03:00
int fd ;
2016-12-09 19:01:37 +03:00
2018-08-22 16:25:26 +03:00
DBG_DEBUG ( " Path [%s] \n " , smb_fname_str_dbg ( smb_fname ) ) ;
2016-12-09 19:01:37 +03:00
2018-08-22 16:25:26 +03:00
SMB_VFS_HANDLE_GET_DATA ( handle , config ,
struct fruit_config_data , return - 1 ) ;
2016-12-09 19:01:37 +03:00
2020-12-08 18:29:10 +03:00
fio = VFS_ADD_FSP_EXTENSION ( handle , fsp , struct fio , fio_destroy_fn ) ;
fio - > handle = handle ;
fio - > fsp = fsp ;
2018-08-22 16:25:26 +03:00
fio - > type = ADOUBLE_META ;
fio - > config = config ;
2016-12-09 19:01:37 +03:00
2020-05-21 00:05:07 +03:00
fd = SMB_VFS_NEXT_OPENAT ( handle ,
dirfsp ,
smb_fname ,
fsp ,
2022-06-03 16:53:29 +03:00
& how ) ;
2018-08-22 16:25:26 +03:00
if ( fd ! = - 1 ) {
return fd ;
2016-12-09 19:01:37 +03:00
}
2018-08-22 16:25:26 +03:00
if ( ! ( flags & O_CREAT ) ) {
VFS_REMOVE_FSP_EXTENSION ( handle , fsp ) ;
return - 1 ;
2016-12-09 19:01:37 +03:00
}
2020-07-23 08:32:11 +03:00
fd = vfs_fake_fd ( ) ;
2018-08-22 16:25:26 +03:00
if ( fd = = - 1 ) {
VFS_REMOVE_FSP_EXTENSION ( handle , fsp ) ;
return - 1 ;
}
2016-12-09 19:01:37 +03:00
2018-08-22 16:25:26 +03:00
fio - > fake_fd = true ;
fio - > flags = flags ;
fio - > mode = mode ;
2016-12-09 19:01:37 +03:00
2018-08-22 16:25:26 +03:00
return fd ;
2016-11-29 18:56:00 +03:00
}
static int fruit_open_meta_netatalk ( vfs_handle_struct * handle ,
2020-05-20 18:51:23 +03:00
const struct files_struct * dirfsp ,
2020-05-21 21:56:14 +03:00
const struct smb_filename * smb_fname ,
2016-11-29 18:56:00 +03:00
files_struct * fsp ,
int flags ,
mode_t mode )
2014-06-23 18:59:45 +04:00
{
2018-08-22 16:25:26 +03:00
struct fruit_config_data * config = NULL ;
struct fio * fio = NULL ;
2014-06-23 18:59:45 +04:00
struct adouble * ad = NULL ;
2018-08-22 16:25:26 +03:00
bool meta_exists = false ;
int fd ;
2014-06-23 18:59:45 +04:00
2016-12-08 22:34:55 +03:00
DBG_DEBUG ( " Path [%s] \n " , smb_fname_str_dbg ( smb_fname ) ) ;
2021-07-01 21:26:21 +03:00
/*
* We know this is a stream open , so fsp - > base_fsp must
* already be open .
*/
2022-02-11 11:37:35 +03:00
SMB_ASSERT ( fsp_is_alternate_stream ( fsp ) ) ;
2021-07-01 21:26:21 +03:00
SMB_ASSERT ( fsp - > base_fsp - > fsp_name - > fsp = = fsp - > base_fsp ) ;
2021-07-02 23:47:30 +03:00
ad = ad_get ( talloc_tos ( ) , handle , fsp - > base_fsp - > fsp_name , ADOUBLE_META ) ;
2018-08-22 16:25:26 +03:00
if ( ad ! = NULL ) {
meta_exists = true ;
2014-06-23 18:59:45 +04:00
}
2018-08-22 16:25:26 +03:00
TALLOC_FREE ( ad ) ;
2016-12-08 21:12:32 +03:00
2018-08-22 16:25:26 +03:00
if ( ! meta_exists & & ! ( flags & O_CREAT ) ) {
errno = ENOENT ;
return - 1 ;
2014-06-23 18:59:45 +04:00
}
2020-07-23 08:32:11 +03:00
fd = vfs_fake_fd ( ) ;
2018-08-22 16:25:26 +03:00
if ( fd = = - 1 ) {
return - 1 ;
2014-06-23 18:59:45 +04:00
}
2018-08-22 16:25:26 +03:00
SMB_VFS_HANDLE_GET_DATA ( handle , config ,
struct fruit_config_data , return - 1 ) ;
2020-12-08 18:29:10 +03:00
fio = VFS_ADD_FSP_EXTENSION ( handle , fsp , struct fio , fio_destroy_fn ) ;
fio - > handle = handle ;
fio - > fsp = fsp ;
2018-08-22 16:25:26 +03:00
fio - > type = ADOUBLE_META ;
fio - > config = config ;
fio - > fake_fd = true ;
fio - > flags = flags ;
fio - > mode = mode ;
return fd ;
2014-06-23 18:59:45 +04:00
}
2016-11-29 18:56:00 +03:00
static int fruit_open_meta ( vfs_handle_struct * handle ,
2020-05-20 18:51:23 +03:00
const struct files_struct * dirfsp ,
2020-05-21 21:56:14 +03:00
const struct smb_filename * smb_fname ,
2016-11-29 18:56:00 +03:00
files_struct * fsp , int flags , mode_t mode )
{
2016-12-08 22:34:55 +03:00
int fd ;
2016-11-29 18:56:00 +03:00
struct fruit_config_data * config = NULL ;
DBG_DEBUG ( " path [%s] \n " , smb_fname_str_dbg ( smb_fname ) ) ;
SMB_VFS_HANDLE_GET_DATA ( handle , config ,
struct fruit_config_data , return - 1 ) ;
switch ( config - > meta ) {
case FRUIT_META_STREAM :
2020-05-20 18:51:23 +03:00
fd = fruit_open_meta_stream ( handle , dirfsp , smb_fname ,
2016-11-29 18:56:00 +03:00
fsp , flags , mode ) ;
break ;
case FRUIT_META_NETATALK :
2020-05-20 18:51:23 +03:00
fd = fruit_open_meta_netatalk ( handle , dirfsp , smb_fname ,
2016-11-29 18:56:00 +03:00
fsp , flags , mode ) ;
break ;
default :
DBG_ERR ( " Unexpected meta config [%d] \n " , config - > meta ) ;
return - 1 ;
}
2016-12-08 22:34:55 +03:00
DBG_DEBUG ( " path [%s] fd [%d] \n " , smb_fname_str_dbg ( smb_fname ) , fd ) ;
return fd ;
2016-11-29 18:56:00 +03:00
}
2016-12-02 12:46:55 +03:00
static int fruit_open_rsrc_adouble ( vfs_handle_struct * handle ,
2020-05-20 18:51:23 +03:00
const struct files_struct * dirfsp ,
2020-05-21 21:56:14 +03:00
const struct smb_filename * smb_fname ,
2016-12-02 12:46:55 +03:00
files_struct * fsp ,
int flags ,
mode_t mode )
2014-06-23 18:59:45 +04:00
{
int rc = 0 ;
2016-12-08 22:34:55 +03:00
struct fruit_config_data * config = NULL ;
2020-12-08 18:29:10 +03:00
struct files_struct * ad_fsp = NULL ;
struct fio * fio = NULL ;
struct fio * ref_fio = NULL ;
NTSTATUS status ;
int fd = - 1 ;
2014-06-23 18:59:45 +04:00
2016-12-08 22:34:55 +03:00
SMB_VFS_HANDLE_GET_DATA ( handle , config ,
struct fruit_config_data , return - 1 ) ;
2017-02-07 17:01:53 +03:00
if ( ( ! ( flags & O_CREAT ) ) & &
S_ISDIR ( fsp - > base_fsp - > fsp_name - > st . st_ex_mode ) )
{
2022-02-22 19:12:34 +03:00
/* sorry, but directories don't have a resource fork */
2023-05-22 20:35:33 +03:00
errno = ENOENT ;
2014-06-23 18:59:45 +04:00
rc = - 1 ;
goto exit ;
}
2020-12-08 18:29:10 +03:00
/*
* We return a fake_fd to the vfs modules above ,
* while we open an internal backend fsp for the
* ' . _ ' file for the next vfs modules .
*
* Note that adouble_open_from_base_fsp ( ) recurses
* into fruit_openat ( ) , but it ' ll just pass to
* the next module as just opens a flat file on
* disk .
*/
fd = vfs_fake_fd ( ) ;
if ( fd = = - 1 ) {
rc = fd ;
2014-06-23 18:59:45 +04:00
goto exit ;
}
2022-02-23 12:30:06 +03:00
status = adouble_open_from_base_fsp ( fsp - > conn - > cwd_fsp ,
2020-12-08 18:29:10 +03:00
fsp - > base_fsp ,
ADOUBLE_RSRC ,
flags ,
mode ,
& ad_fsp ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
errno = map_errno_from_nt_status ( status ) ;
2014-06-23 18:59:45 +04:00
rc = - 1 ;
goto exit ;
}
2020-12-08 18:29:10 +03:00
/*
* Now we need to glue both handles together ,
* so that they automatically detach each other
* on close .
*/
fio = fruit_get_complete_fio ( handle , fsp ) ;
2022-01-10 15:26:25 +03:00
if ( fio = = NULL ) {
DBG_ERR ( " fio=NULL for [%s] \n " , fsp_str_dbg ( fsp ) ) ;
errno = EBADF ;
rc = - 1 ;
goto exit ;
}
2016-12-08 21:12:32 +03:00
2020-12-08 18:29:10 +03:00
ref_fio = VFS_ADD_FSP_EXTENSION ( handle , ad_fsp ,
struct fio ,
fio_ref_destroy_fn ) ;
if ( ref_fio = = NULL ) {
int saved_errno = errno ;
fd_close ( ad_fsp ) ;
file_free ( NULL , ad_fsp ) ;
ad_fsp = NULL ;
errno = saved_errno ;
rc = - 1 ;
goto exit ;
2014-06-23 18:59:45 +04:00
}
2020-12-08 18:29:10 +03:00
SMB_ASSERT ( ref_fio - > fsp = = NULL ) ;
ref_fio - > handle = handle ;
ref_fio - > fsp = ad_fsp ;
ref_fio - > type = ADOUBLE_RSRC ;
ref_fio - > config = config ;
ref_fio - > real_fio = fio ;
SMB_ASSERT ( fio - > ad_fsp = = NULL ) ;
fio - > ad_fsp = ad_fsp ;
fio - > fake_fd = true ;
2014-06-23 18:59:45 +04:00
2020-12-08 18:29:10 +03:00
exit :
2014-06-23 18:59:45 +04:00
2020-12-08 18:29:10 +03:00
DEBUG ( 10 , ( " fruit_open resource fork: rc=%d \n " , rc ) ) ;
2014-06-23 18:59:45 +04:00
if ( rc ! = 0 ) {
int saved_errno = errno ;
2020-12-08 18:29:10 +03:00
if ( fd ! = - 1 ) {
vfs_fake_fd_close ( fd ) ;
2014-06-23 18:59:45 +04:00
}
errno = saved_errno ;
2020-12-08 18:29:10 +03:00
return rc ;
2014-06-23 18:59:45 +04:00
}
2020-12-08 18:29:10 +03:00
return fd ;
2014-06-23 18:59:45 +04:00
}
2016-12-02 12:46:55 +03:00
static int fruit_open_rsrc_xattr ( vfs_handle_struct * handle ,
2020-05-20 18:51:23 +03:00
const struct files_struct * dirfsp ,
2020-05-21 21:56:14 +03:00
const struct smb_filename * smb_fname ,
2016-12-02 12:46:55 +03:00
files_struct * fsp ,
int flags ,
mode_t mode )
{
# ifdef HAVE_ATTROPEN
int fd = - 1 ;
2020-05-21 00:05:07 +03:00
/*
* As there ' s no attropenat ( ) this is only going to work with AT_FDCWD .
*/
2020-10-17 18:01:47 +03:00
SMB_ASSERT ( fsp_get_pathref_fd ( dirfsp ) = = AT_FDCWD ) ;
2020-05-21 00:05:07 +03:00
2016-12-02 12:46:55 +03:00
fd = attropen ( smb_fname - > base_name ,
2016-12-08 22:34:55 +03:00
AFPRESOURCE_EA_NETATALK ,
flags ,
mode ) ;
2016-12-02 12:46:55 +03:00
if ( fd = = - 1 ) {
return - 1 ;
}
return fd ;
# else
errno = ENOSYS ;
return - 1 ;
# endif
}
static int fruit_open_rsrc ( vfs_handle_struct * handle ,
2020-05-20 18:51:23 +03:00
const struct files_struct * dirfsp ,
2020-05-21 21:56:14 +03:00
const struct smb_filename * smb_fname ,
2016-12-02 12:46:55 +03:00
files_struct * fsp , int flags , mode_t mode )
{
int fd ;
struct fruit_config_data * config = NULL ;
2016-12-08 22:34:55 +03:00
struct fio * fio = NULL ;
2016-12-02 12:46:55 +03:00
DBG_DEBUG ( " Path [%s] \n " , smb_fname_str_dbg ( smb_fname ) ) ;
SMB_VFS_HANDLE_GET_DATA ( handle , config ,
struct fruit_config_data , return - 1 ) ;
2020-12-08 18:29:10 +03:00
fio = VFS_ADD_FSP_EXTENSION ( handle , fsp , struct fio , fio_destroy_fn ) ;
fio - > handle = handle ;
fio - > fsp = fsp ;
fio - > type = ADOUBLE_RSRC ;
fio - > config = config ;
2016-12-02 12:46:55 +03:00
switch ( config - > rsrc ) {
2022-06-03 16:53:29 +03:00
case FRUIT_RSRC_STREAM : {
struct vfs_open_how how = {
. flags = flags , . mode = mode ,
} ;
2020-05-21 00:05:07 +03:00
fd = SMB_VFS_NEXT_OPENAT ( handle ,
dirfsp ,
smb_fname ,
fsp ,
2022-06-03 16:53:29 +03:00
& how ) ;
2016-12-02 12:46:55 +03:00
break ;
2022-06-03 16:53:29 +03:00
}
2016-12-02 12:46:55 +03:00
case FRUIT_RSRC_ADFILE :
2020-05-20 18:51:23 +03:00
fd = fruit_open_rsrc_adouble ( handle , dirfsp , smb_fname ,
2016-12-02 12:46:55 +03:00
fsp , flags , mode ) ;
break ;
case FRUIT_RSRC_XATTR :
2020-05-20 18:51:23 +03:00
fd = fruit_open_rsrc_xattr ( handle , dirfsp , smb_fname ,
2016-12-02 12:46:55 +03:00
fsp , flags , mode ) ;
break ;
default :
DBG_ERR ( " Unexpected rsrc config [%d] \n " , config - > rsrc ) ;
2022-02-22 19:12:44 +03:00
errno = EINVAL ;
2016-12-02 12:46:55 +03:00
return - 1 ;
}
DBG_DEBUG ( " Path [%s] fd [%d] \n " , smb_fname_str_dbg ( smb_fname ) , fd ) ;
2016-12-08 22:34:55 +03:00
if ( fd = = - 1 ) {
return - 1 ;
}
2016-12-02 12:46:55 +03:00
return fd ;
}
2020-05-21 00:04:26 +03:00
static int fruit_openat ( vfs_handle_struct * handle ,
const struct files_struct * dirfsp ,
2020-05-21 21:56:14 +03:00
const struct smb_filename * smb_fname ,
2020-05-21 00:04:26 +03:00
files_struct * fsp ,
2022-06-03 16:53:29 +03:00
const struct vfs_open_how * how )
2020-05-21 00:04:26 +03:00
{
int fd ;
DBG_DEBUG ( " Path [%s] \n " , smb_fname_str_dbg ( smb_fname ) ) ;
if ( ! is_named_stream ( smb_fname ) ) {
return SMB_VFS_NEXT_OPENAT ( handle ,
dirfsp ,
smb_fname ,
fsp ,
2022-06-03 16:53:29 +03:00
how ) ;
2020-05-21 00:04:26 +03:00
}
2022-06-03 17:45:41 +03:00
if ( how - > resolve ! = 0 ) {
errno = ENOSYS ;
return - 1 ;
}
2022-03-31 18:40:35 +03:00
SMB_ASSERT ( fsp_is_alternate_stream ( fsp ) ) ;
2020-05-21 00:04:26 +03:00
if ( is_afpinfo_stream ( smb_fname - > stream_name ) ) {
fd = fruit_open_meta ( handle ,
dirfsp ,
smb_fname ,
fsp ,
2022-06-03 16:53:29 +03:00
how - > flags ,
how - > mode ) ;
2020-05-21 00:04:26 +03:00
} else if ( is_afpresource_stream ( smb_fname - > stream_name ) ) {
fd = fruit_open_rsrc ( handle ,
dirfsp ,
smb_fname ,
fsp ,
2022-06-03 16:53:29 +03:00
how - > flags ,
how - > mode ) ;
2020-05-21 00:04:26 +03:00
} else {
fd = SMB_VFS_NEXT_OPENAT ( handle ,
dirfsp ,
smb_fname ,
fsp ,
2022-06-03 16:53:29 +03:00
how ) ;
2020-05-21 00:04:26 +03:00
}
DBG_DEBUG ( " Path [%s] fd [%d] \n " , smb_fname_str_dbg ( smb_fname ) , fd ) ;
2020-11-24 20:02:26 +03:00
/* Prevent reopen optimisation */
fsp - > fsp_flags . have_proc_fds = false ;
2020-05-21 00:04:26 +03:00
return fd ;
}
2018-12-18 19:18:33 +03:00
static int fruit_close_meta ( vfs_handle_struct * handle ,
files_struct * fsp )
{
int ret ;
struct fruit_config_data * config = NULL ;
SMB_VFS_HANDLE_GET_DATA ( handle , config ,
struct fruit_config_data , return - 1 ) ;
switch ( config - > meta ) {
case FRUIT_META_STREAM :
2022-01-10 15:26:25 +03:00
{
struct fio * fio = fruit_get_complete_fio ( handle , fsp ) ;
if ( fio = = NULL ) {
return - 1 ;
}
2020-12-11 14:59:28 +03:00
if ( fio - > fake_fd ) {
ret = vfs_fake_fd_close ( fsp_get_pathref_fd ( fsp ) ) ;
fsp_set_fd ( fsp , - 1 ) ;
} else {
ret = SMB_VFS_NEXT_CLOSE ( handle , fsp ) ;
}
2018-12-18 19:18:33 +03:00
break ;
2022-01-10 15:26:25 +03:00
}
2018-12-18 19:18:33 +03:00
case FRUIT_META_NETATALK :
2020-12-18 16:36:00 +03:00
ret = vfs_fake_fd_close ( fsp_get_pathref_fd ( fsp ) ) ;
2020-09-26 22:46:51 +03:00
fsp_set_fd ( fsp , - 1 ) ;
2018-12-18 19:18:33 +03:00
break ;
default :
DBG_ERR ( " Unexpected meta config [%d] \n " , config - > meta ) ;
return - 1 ;
}
return ret ;
}
static int fruit_close_rsrc ( vfs_handle_struct * handle ,
files_struct * fsp )
{
int ret ;
struct fruit_config_data * config = NULL ;
SMB_VFS_HANDLE_GET_DATA ( handle , config ,
struct fruit_config_data , return - 1 ) ;
switch ( config - > rsrc ) {
case FRUIT_RSRC_STREAM :
ret = SMB_VFS_NEXT_CLOSE ( handle , fsp ) ;
break ;
2020-12-08 18:29:10 +03:00
case FRUIT_RSRC_ADFILE :
2022-01-10 15:26:25 +03:00
{
struct fio * fio = fruit_get_complete_fio ( handle , fsp ) ;
if ( fio = = NULL ) {
return - 1 ;
}
2020-12-08 18:29:10 +03:00
fio_close_ad_fsp ( fio ) ;
ret = vfs_fake_fd_close ( fsp_get_pathref_fd ( fsp ) ) ;
fsp_set_fd ( fsp , - 1 ) ;
break ;
2022-01-10 15:26:25 +03:00
}
2020-12-08 18:29:10 +03:00
2018-12-18 19:18:33 +03:00
case FRUIT_RSRC_XATTR :
2020-12-18 16:36:00 +03:00
ret = vfs_fake_fd_close ( fsp_get_pathref_fd ( fsp ) ) ;
2020-09-26 22:46:51 +03:00
fsp_set_fd ( fsp , - 1 ) ;
2018-12-18 19:18:33 +03:00
break ;
default :
DBG_ERR ( " Unexpected rsrc config [%d] \n " , config - > rsrc ) ;
return - 1 ;
}
return ret ;
}
static int fruit_close ( vfs_handle_struct * handle ,
files_struct * fsp )
{
int ret ;
int fd ;
2020-10-03 22:24:29 +03:00
fd = fsp_get_pathref_fd ( fsp ) ;
2018-12-18 19:18:33 +03:00
DBG_DEBUG ( " Path [%s] fd [%d] \n " , smb_fname_str_dbg ( fsp - > fsp_name ) , fd ) ;
2022-02-11 11:59:16 +03:00
if ( ! fsp_is_alternate_stream ( fsp ) ) {
2018-12-18 19:18:33 +03:00
return SMB_VFS_NEXT_CLOSE ( handle , fsp ) ;
}
2019-07-01 16:53:36 +03:00
if ( is_afpinfo_stream ( fsp - > fsp_name - > stream_name ) ) {
2018-12-18 19:18:33 +03:00
ret = fruit_close_meta ( handle , fsp ) ;
2019-07-01 16:53:36 +03:00
} else if ( is_afpresource_stream ( fsp - > fsp_name - > stream_name ) ) {
2018-12-18 19:18:33 +03:00
ret = fruit_close_rsrc ( handle , fsp ) ;
} else {
ret = SMB_VFS_NEXT_CLOSE ( handle , fsp ) ;
}
return ret ;
}
2019-08-10 00:22:03 +03:00
static int fruit_renameat ( struct vfs_handle_struct * handle ,
files_struct * srcfsp ,
const struct smb_filename * smb_fname_src ,
files_struct * dstfsp ,
2024-08-06 14:21:34 +03:00
const struct smb_filename * smb_fname_dst ,
const struct vfs_rename_how * how )
2019-08-10 00:22:03 +03:00
{
int rc = - 1 ;
struct fruit_config_data * config = NULL ;
struct smb_filename * src_adp_smb_fname = NULL ;
struct smb_filename * dst_adp_smb_fname = NULL ;
SMB_VFS_HANDLE_GET_DATA ( handle , config ,
struct fruit_config_data , return - 1 ) ;
if ( ! VALID_STAT ( smb_fname_src - > st ) ) {
DBG_ERR ( " Need valid stat for [%s] \n " ,
smb_fname_str_dbg ( smb_fname_src ) ) ;
return - 1 ;
}
rc = SMB_VFS_NEXT_RENAMEAT ( handle ,
srcfsp ,
smb_fname_src ,
dstfsp ,
2024-08-06 14:21:34 +03:00
smb_fname_dst ,
how ) ;
2019-08-10 00:22:03 +03:00
if ( rc ! = 0 ) {
return - 1 ;
}
if ( ( config - > rsrc ! = FRUIT_RSRC_ADFILE ) | |
( ! S_ISREG ( smb_fname_src - > st . st_ex_mode ) ) )
{
return 0 ;
}
rc = adouble_path ( talloc_tos ( ) , smb_fname_src , & src_adp_smb_fname ) ;
if ( rc ! = 0 ) {
goto done ;
}
rc = adouble_path ( talloc_tos ( ) , smb_fname_dst , & dst_adp_smb_fname ) ;
if ( rc ! = 0 ) {
goto done ;
}
DBG_DEBUG ( " %s -> %s \n " ,
smb_fname_str_dbg ( src_adp_smb_fname ) ,
smb_fname_str_dbg ( dst_adp_smb_fname ) ) ;
rc = SMB_VFS_NEXT_RENAMEAT ( handle ,
srcfsp ,
src_adp_smb_fname ,
dstfsp ,
2024-08-06 14:21:34 +03:00
dst_adp_smb_fname ,
how ) ;
2019-08-10 00:22:03 +03:00
if ( errno = = ENOENT ) {
rc = 0 ;
}
done :
TALLOC_FREE ( src_adp_smb_fname ) ;
TALLOC_FREE ( dst_adp_smb_fname ) ;
return rc ;
}
2016-12-02 11:00:31 +03:00
static int fruit_unlink_meta_stream ( vfs_handle_struct * handle ,
2019-09-13 23:46:00 +03:00
struct files_struct * dirfsp ,
const struct smb_filename * smb_fname )
2016-12-02 11:00:31 +03:00
{
2019-09-13 23:46:00 +03:00
return SMB_VFS_NEXT_UNLINKAT ( handle ,
dirfsp ,
smb_fname ,
0 ) ;
2016-12-02 11:00:31 +03:00
}
static int fruit_unlink_meta_netatalk ( vfs_handle_struct * handle ,
const struct smb_filename * smb_fname )
{
2021-01-21 12:27:22 +03:00
SMB_ASSERT ( smb_fname - > fsp ! = NULL ) ;
2022-02-11 11:37:35 +03:00
SMB_ASSERT ( fsp_is_alternate_stream ( smb_fname - > fsp ) ) ;
2021-01-21 12:27:22 +03:00
return SMB_VFS_FREMOVEXATTR ( smb_fname - > fsp - > base_fsp ,
2016-12-02 11:00:31 +03:00
AFPINFO_EA_NETATALK ) ;
}
static int fruit_unlink_meta ( vfs_handle_struct * handle ,
2019-09-13 23:44:21 +03:00
struct files_struct * dirfsp ,
const struct smb_filename * smb_fname )
2014-06-23 18:59:45 +04:00
{
struct fruit_config_data * config = NULL ;
2016-12-02 11:00:31 +03:00
int rc ;
2014-06-23 18:59:45 +04:00
SMB_VFS_HANDLE_GET_DATA ( handle , config ,
struct fruit_config_data , return - 1 ) ;
2016-12-02 11:00:31 +03:00
switch ( config - > meta ) {
case FRUIT_META_STREAM :
2019-09-13 23:46:00 +03:00
rc = fruit_unlink_meta_stream ( handle ,
dirfsp ,
smb_fname ) ;
2016-12-02 11:00:31 +03:00
break ;
2015-08-25 18:06:52 +03:00
2016-12-02 11:00:31 +03:00
case FRUIT_META_NETATALK :
rc = fruit_unlink_meta_netatalk ( handle , smb_fname ) ;
break ;
2015-08-25 18:06:52 +03:00
2016-12-02 11:00:31 +03:00
default :
DBG_ERR ( " Unsupported meta config [%d] \n " , config - > meta ) ;
return - 1 ;
}
return rc ;
}
static int fruit_unlink_rsrc_stream ( vfs_handle_struct * handle ,
2019-09-14 00:10:16 +03:00
struct files_struct * dirfsp ,
const struct smb_filename * smb_fname ,
bool force_unlink )
2016-12-02 11:00:31 +03:00
{
int ret ;
if ( ! force_unlink ) {
2021-01-21 12:41:19 +03:00
struct smb_filename * full_fname = NULL ;
2016-12-02 11:00:31 +03:00
off_t size ;
2021-01-21 12:41:19 +03:00
/*
* TODO : use SMB_VFS_STATX ( ) once we have it .
*/
full_fname = full_path_from_dirfsp_atname ( talloc_tos ( ) ,
dirfsp ,
smb_fname ) ;
if ( full_fname = = NULL ) {
2016-12-02 11:00:31 +03:00
return - 1 ;
2015-08-25 18:06:52 +03:00
}
/*
* 0 byte resource fork streams are not listed by
* vfs_streaminfo , as a result stream cleanup / deletion of file
* deletion doesn ' t remove the resourcefork stream .
*/
2016-12-02 11:00:31 +03:00
2021-01-21 12:41:19 +03:00
ret = SMB_VFS_NEXT_STAT ( handle , full_fname ) ;
2016-12-02 11:00:31 +03:00
if ( ret ! = 0 ) {
2021-01-21 12:41:19 +03:00
TALLOC_FREE ( full_fname ) ;
2016-12-02 11:00:31 +03:00
DBG_ERR ( " stat [%s] failed [%s] \n " ,
2021-01-21 12:41:19 +03:00
smb_fname_str_dbg ( full_fname ) , strerror ( errno ) ) ;
2015-08-25 18:06:52 +03:00
return - 1 ;
}
2021-01-21 12:41:19 +03:00
size = full_fname - > st . st_ex_size ;
TALLOC_FREE ( full_fname ) ;
2016-12-02 11:00:31 +03:00
if ( size > 0 ) {
/* OS X ignores resource fork stream delete requests */
return 0 ;
2015-08-25 18:06:52 +03:00
}
2016-12-02 11:00:31 +03:00
}
2015-08-25 18:06:52 +03:00
2019-09-14 00:10:16 +03:00
ret = SMB_VFS_NEXT_UNLINKAT ( handle ,
dirfsp ,
smb_fname ,
0 ) ;
2016-12-02 11:00:31 +03:00
if ( ( ret ! = 0 ) & & ( errno = = ENOENT ) & & force_unlink ) {
ret = 0 ;
2015-08-25 18:06:52 +03:00
}
2016-12-02 11:00:31 +03:00
return ret ;
}
static int fruit_unlink_rsrc_adouble ( vfs_handle_struct * handle ,
2019-09-14 00:11:46 +03:00
struct files_struct * dirfsp ,
const struct smb_filename * smb_fname ,
bool force_unlink )
2016-12-02 11:00:31 +03:00
{
int rc ;
struct adouble * ad = NULL ;
struct smb_filename * adp_smb_fname = NULL ;
if ( ! force_unlink ) {
2021-01-21 12:41:41 +03:00
struct smb_filename * full_fname = NULL ;
full_fname = full_path_from_dirfsp_atname ( talloc_tos ( ) ,
dirfsp ,
smb_fname ) ;
if ( full_fname = = NULL ) {
return - 1 ;
}
ad = ad_get ( talloc_tos ( ) , handle , full_fname ,
2016-12-02 11:00:31 +03:00
ADOUBLE_RSRC ) ;
2021-01-21 12:41:41 +03:00
TALLOC_FREE ( full_fname ) ;
2016-12-02 11:00:31 +03:00
if ( ad = = NULL ) {
errno = ENOENT ;
return - 1 ;
2014-06-23 18:59:45 +04:00
}
2015-08-25 18:06:52 +03:00
2016-12-02 11:00:31 +03:00
/*
* 0 byte resource fork streams are not listed by
* vfs_streaminfo , as a result stream cleanup / deletion of file
* deletion doesn ' t remove the resourcefork stream .
*/
if ( ad_getentrylen ( ad , ADEID_RFORK ) > 0 ) {
/* OS X ignores resource fork stream delete requests */
TALLOC_FREE ( ad ) ;
return 0 ;
}
TALLOC_FREE ( ad ) ;
2015-08-25 18:06:52 +03:00
}
2017-05-25 21:38:26 +03:00
rc = adouble_path ( talloc_tos ( ) , smb_fname , & adp_smb_fname ) ;
2016-12-02 11:00:31 +03:00
if ( rc ! = 0 ) {
return - 1 ;
2014-06-23 18:59:45 +04:00
}
2019-09-14 00:11:46 +03:00
rc = SMB_VFS_NEXT_UNLINKAT ( handle ,
dirfsp ,
adp_smb_fname ,
0 ) ;
2016-12-02 11:00:31 +03:00
TALLOC_FREE ( adp_smb_fname ) ;
2023-10-26 11:15:39 +03:00
if ( ( rc ! = 0 ) & & ( errno = = ENOENT | | errno = = ENAMETOOLONG ) & & force_unlink ) {
2016-12-02 11:00:31 +03:00
rc = 0 ;
}
2015-08-25 18:06:52 +03:00
2016-12-02 11:00:31 +03:00
return rc ;
}
2015-08-25 18:06:52 +03:00
2016-12-02 11:00:31 +03:00
static int fruit_unlink_rsrc_xattr ( vfs_handle_struct * handle ,
const struct smb_filename * smb_fname ,
bool force_unlink )
{
/*
* OS X ignores resource fork stream delete requests , so nothing to do
* here . Removing the file will remove the xattr anyway , so we don ' t
* have to take care of removing 0 byte resource forks that could be
* left behind .
*/
2015-08-25 18:06:52 +03:00
return 0 ;
2014-06-23 18:59:45 +04:00
}
2016-12-02 11:00:31 +03:00
static int fruit_unlink_rsrc ( vfs_handle_struct * handle ,
2019-09-14 00:08:08 +03:00
struct files_struct * dirfsp ,
const struct smb_filename * smb_fname ,
bool force_unlink )
2016-12-02 11:00:31 +03:00
{
struct fruit_config_data * config = NULL ;
int rc ;
SMB_VFS_HANDLE_GET_DATA ( handle , config ,
struct fruit_config_data , return - 1 ) ;
switch ( config - > rsrc ) {
case FRUIT_RSRC_STREAM :
2019-09-14 00:10:16 +03:00
rc = fruit_unlink_rsrc_stream ( handle ,
dirfsp ,
smb_fname ,
force_unlink ) ;
2016-12-02 11:00:31 +03:00
break ;
case FRUIT_RSRC_ADFILE :
2019-09-14 00:11:46 +03:00
rc = fruit_unlink_rsrc_adouble ( handle ,
dirfsp ,
smb_fname ,
force_unlink ) ;
2016-12-02 11:00:31 +03:00
break ;
case FRUIT_RSRC_XATTR :
rc = fruit_unlink_rsrc_xattr ( handle , smb_fname , force_unlink ) ;
break ;
default :
DBG_ERR ( " Unsupported rsrc config [%d] \n " , config - > rsrc ) ;
return - 1 ;
}
return rc ;
}
2021-04-09 16:53:20 +03:00
static int fruit_fchmod ( vfs_handle_struct * handle ,
struct files_struct * fsp ,
mode_t mode )
{
int rc = - 1 ;
struct fruit_config_data * config = NULL ;
struct smb_filename * smb_fname_adp = NULL ;
const struct smb_filename * smb_fname = NULL ;
NTSTATUS status ;
rc = SMB_VFS_NEXT_FCHMOD ( handle , fsp , mode ) ;
if ( rc ! = 0 ) {
return rc ;
}
smb_fname = fsp - > fsp_name ;
SMB_VFS_HANDLE_GET_DATA ( handle , config ,
struct fruit_config_data , return - 1 ) ;
if ( config - > rsrc ! = FRUIT_RSRC_ADFILE ) {
return 0 ;
}
if ( ! VALID_STAT ( smb_fname - > st ) ) {
return 0 ;
}
if ( ! S_ISREG ( smb_fname - > st . st_ex_mode ) ) {
return 0 ;
}
rc = adouble_path ( talloc_tos ( ) , smb_fname , & smb_fname_adp ) ;
if ( rc ! = 0 ) {
return - 1 ;
}
status = openat_pathref_fsp ( handle - > conn - > cwd_fsp ,
smb_fname_adp ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
/* detect ENOENT (mapped to OBJECT_NAME_NOT_FOUND) */
if ( NT_STATUS_EQUAL ( status ,
NT_STATUS_OBJECT_NAME_NOT_FOUND ) ) {
rc = 0 ;
goto out ;
}
rc = - 1 ;
goto out ;
}
DBG_DEBUG ( " %s \n " , smb_fname_adp - > base_name ) ;
rc = SMB_VFS_NEXT_FCHMOD ( handle , smb_fname_adp - > fsp , mode ) ;
if ( errno = = ENOENT ) {
rc = 0 ;
}
out :
TALLOC_FREE ( smb_fname_adp ) ;
return rc ;
}
2019-09-12 20:46:02 +03:00
static int fruit_unlinkat ( vfs_handle_struct * handle ,
struct files_struct * dirfsp ,
const struct smb_filename * smb_fname ,
int flags )
{
2020-03-19 16:07:23 +03:00
struct fruit_config_data * config = NULL ;
struct smb_filename * rsrc_smb_fname = NULL ;
int ret ;
2019-09-12 20:46:02 +03:00
if ( flags & AT_REMOVEDIR ) {
2020-03-17 21:03:53 +03:00
return SMB_VFS_NEXT_UNLINKAT ( handle ,
dirfsp ,
smb_fname ,
AT_REMOVEDIR ) ;
2019-09-12 20:46:02 +03:00
}
2020-03-19 16:07:23 +03:00
SMB_VFS_HANDLE_GET_DATA ( handle , config ,
struct fruit_config_data , return - 1 ) ;
if ( is_afpinfo_stream ( smb_fname - > stream_name ) ) {
return fruit_unlink_meta ( handle ,
dirfsp ,
smb_fname ) ;
} else if ( is_afpresource_stream ( smb_fname - > stream_name ) ) {
return fruit_unlink_rsrc ( handle ,
dirfsp ,
smb_fname ,
false ) ;
} else if ( is_named_stream ( smb_fname ) ) {
return SMB_VFS_NEXT_UNLINKAT ( handle ,
dirfsp ,
smb_fname ,
0 ) ;
} else if ( is_adouble_file ( smb_fname - > base_name ) ) {
return SMB_VFS_NEXT_UNLINKAT ( handle ,
dirfsp ,
smb_fname ,
0 ) ;
}
/*
* A request to delete the base file . Because 0 byte resource
* fork streams are not listed by fruit_streaminfo ,
* delete_all_streams ( ) can ' t remove 0 byte resource fork
* streams , so we have to cleanup this here .
*/
rsrc_smb_fname = synthetic_smb_fname ( talloc_tos ( ) ,
smb_fname - > base_name ,
AFPRESOURCE_STREAM_NAME ,
NULL ,
2020-04-30 12:48:32 +03:00
smb_fname - > twrp ,
2020-03-19 16:07:23 +03:00
smb_fname - > flags ) ;
if ( rsrc_smb_fname = = NULL ) {
return - 1 ;
}
ret = fruit_unlink_rsrc ( handle , dirfsp , rsrc_smb_fname , true ) ;
if ( ( ret ! = 0 ) & & ( errno ! = ENOENT ) ) {
DBG_ERR ( " Forced unlink of [%s] failed [%s] \n " ,
smb_fname_str_dbg ( rsrc_smb_fname ) , strerror ( errno ) ) ;
TALLOC_FREE ( rsrc_smb_fname ) ;
return - 1 ;
}
TALLOC_FREE ( rsrc_smb_fname ) ;
return SMB_VFS_NEXT_UNLINKAT ( handle ,
dirfsp ,
smb_fname ,
0 ) ;
2019-09-12 20:46:02 +03:00
}
2016-12-08 22:38:17 +03:00
static ssize_t fruit_pread_meta_stream ( vfs_handle_struct * handle ,
files_struct * fsp , void * data ,
size_t n , off_t offset )
{
2020-12-10 15:11:06 +03:00
struct fio * fio = fruit_get_complete_fio ( handle , fsp ) ;
2016-12-11 21:10:05 +03:00
ssize_t nread ;
int ret ;
2021-01-12 13:22:57 +03:00
if ( ( fio = = NULL ) | | fio - > fake_fd ) {
2020-12-11 15:00:56 +03:00
return - 1 ;
}
2016-12-11 21:10:05 +03:00
nread = SMB_VFS_NEXT_PREAD ( handle , fsp , data , n , offset ) ;
2018-08-22 16:22:08 +03:00
if ( nread = = - 1 | | nread = = n ) {
2016-12-11 21:10:05 +03:00
return nread ;
}
DBG_ERR ( " Removing [%s] after short read [%zd] \n " ,
fsp_str_dbg ( fsp ) , nread ) ;
2019-09-14 00:19:21 +03:00
ret = SMB_VFS_NEXT_UNLINKAT ( handle ,
fsp - > conn - > cwd_fsp ,
fsp - > fsp_name ,
0 ) ;
2016-12-11 21:10:05 +03:00
if ( ret ! = 0 ) {
DBG_ERR ( " Removing [%s] failed \n " , fsp_str_dbg ( fsp ) ) ;
return - 1 ;
}
errno = EINVAL ;
return - 1 ;
2016-12-08 22:38:17 +03:00
}
static ssize_t fruit_pread_meta_adouble ( vfs_handle_struct * handle ,
files_struct * fsp , void * data ,
size_t n , off_t offset )
2014-06-23 18:59:45 +04:00
{
AfpInfo * ai = NULL ;
2016-12-08 22:38:17 +03:00
struct adouble * ad = NULL ;
char afpinfo_buf [ AFP_INFO_SIZE ] ;
char * p = NULL ;
ssize_t nread ;
2014-06-23 18:59:45 +04:00
2016-12-08 22:38:17 +03:00
ai = afpinfo_new ( talloc_tos ( ) ) ;
if ( ai = = NULL ) {
return - 1 ;
}
2014-06-23 18:59:45 +04:00
2016-12-08 22:38:17 +03:00
ad = ad_fget ( talloc_tos ( ) , handle , fsp , ADOUBLE_META ) ;
if ( ad = = NULL ) {
nread = - 1 ;
goto fail ;
2014-06-23 18:59:45 +04:00
}
2016-12-08 22:38:17 +03:00
p = ad_get_entry ( ad , ADEID_FINDERI ) ;
if ( p = = NULL ) {
DBG_ERR ( " No ADEID_FINDERI for [%s] \n " , fsp_str_dbg ( fsp ) ) ;
nread = - 1 ;
goto fail ;
}
2014-06-23 18:59:45 +04:00
2016-12-08 22:38:17 +03:00
memcpy ( & ai - > afpi_FinderInfo [ 0 ] , p , ADEDLEN_FINDERI ) ;
nread = afpinfo_pack ( ai , afpinfo_buf ) ;
if ( nread ! = AFP_INFO_SIZE ) {
nread = - 1 ;
goto fail ;
}
2016-11-16 11:34:13 +03:00
2016-12-08 22:38:17 +03:00
memcpy ( data , afpinfo_buf , n ) ;
nread = n ;
2016-11-16 11:34:13 +03:00
2016-12-08 22:38:17 +03:00
fail :
TALLOC_FREE ( ai ) ;
return nread ;
}
static ssize_t fruit_pread_meta ( vfs_handle_struct * handle ,
files_struct * fsp , void * data ,
size_t n , off_t offset )
{
2020-12-10 15:11:06 +03:00
struct fio * fio = fruit_get_complete_fio ( handle , fsp ) ;
2016-12-08 22:38:17 +03:00
ssize_t nread ;
ssize_t to_return ;
/*
* OS X has a off - by - 1 error in the offset calculation , so we ' re
* bug compatible here . It won ' t hurt , as any relevant real
* world read requests from the AFP_AfpInfo stream will be
* offset = 0 n = 60. offset is ignored anyway , see below .
*/
if ( ( offset < 0 ) | | ( offset > = AFP_INFO_SIZE + 1 ) ) {
return 0 ;
2016-11-16 11:34:13 +03:00
}
2018-05-25 13:45:35 +03:00
if ( fio = = NULL ) {
2023-08-07 07:37:51 +03:00
DBG_ERR ( " Failed to fetch fsp extension \n " ) ;
2018-05-25 13:45:35 +03:00
return - 1 ;
}
2016-12-08 22:38:17 +03:00
/* Yes, macOS always reads from offset 0 */
offset = 0 ;
to_return = MIN ( n , AFP_INFO_SIZE ) ;
switch ( fio - > config - > meta ) {
case FRUIT_META_STREAM :
nread = fruit_pread_meta_stream ( handle , fsp , data ,
to_return , offset ) ;
break ;
case FRUIT_META_NETATALK :
nread = fruit_pread_meta_adouble ( handle , fsp , data ,
to_return , offset ) ;
break ;
default :
DBG_ERR ( " Unexpected meta config [%d] \n " , fio - > config - > meta ) ;
return - 1 ;
}
2020-12-11 15:00:09 +03:00
if ( nread = = - 1 & & fio - > fake_fd ) {
2018-08-22 16:22:08 +03:00
AfpInfo * ai = NULL ;
char afpinfo_buf [ AFP_INFO_SIZE ] ;
ai = afpinfo_new ( talloc_tos ( ) ) ;
if ( ai = = NULL ) {
return - 1 ;
}
nread = afpinfo_pack ( ai , afpinfo_buf ) ;
TALLOC_FREE ( ai ) ;
if ( nread ! = AFP_INFO_SIZE ) {
return - 1 ;
}
memcpy ( data , afpinfo_buf , to_return ) ;
return to_return ;
}
2016-12-08 22:38:17 +03:00
return nread ;
}
static ssize_t fruit_pread_rsrc_stream ( vfs_handle_struct * handle ,
files_struct * fsp , void * data ,
size_t n , off_t offset )
{
return SMB_VFS_NEXT_PREAD ( handle , fsp , data , n , offset ) ;
}
static ssize_t fruit_pread_rsrc_xattr ( vfs_handle_struct * handle ,
files_struct * fsp , void * data ,
size_t n , off_t offset )
{
return SMB_VFS_NEXT_PREAD ( handle , fsp , data , n , offset ) ;
}
static ssize_t fruit_pread_rsrc_adouble ( vfs_handle_struct * handle ,
files_struct * fsp , void * data ,
size_t n , off_t offset )
{
2020-12-08 18:29:10 +03:00
struct fio * fio = fruit_get_complete_fio ( handle , fsp ) ;
2016-12-08 22:38:17 +03:00
struct adouble * ad = NULL ;
ssize_t nread ;
2022-01-10 15:26:25 +03:00
if ( fio = = NULL | | fio - > ad_fsp = = NULL ) {
DBG_ERR ( " fio/ad_fsp=NULL for [%s] \n " , fsp_str_dbg ( fsp ) ) ;
2020-12-08 18:29:10 +03:00
errno = EBADF ;
return - 1 ;
}
ad = ad_fget ( talloc_tos ( ) , handle , fio - > ad_fsp , ADOUBLE_RSRC ) ;
2014-06-23 18:59:45 +04:00
if ( ad = = NULL ) {
2020-12-08 18:29:10 +03:00
DBG_ERR ( " ad_fget [%s] failed [%s] \n " ,
fsp_str_dbg ( fio - > ad_fsp ) , strerror ( errno ) ) ;
2016-12-08 22:38:17 +03:00
return - 1 ;
2014-06-23 18:59:45 +04:00
}
2020-12-08 18:29:10 +03:00
nread = SMB_VFS_NEXT_PREAD ( handle , fio - > ad_fsp , data , n ,
2016-12-08 22:38:17 +03:00
offset + ad_getentryoff ( ad , ADEID_RFORK ) ) ;
2015-06-25 16:42:04 +03:00
2016-12-08 22:38:17 +03:00
TALLOC_FREE ( ad ) ;
return nread ;
}
2014-06-23 18:59:45 +04:00
2016-12-08 22:38:17 +03:00
static ssize_t fruit_pread_rsrc ( vfs_handle_struct * handle ,
files_struct * fsp , void * data ,
size_t n , off_t offset )
{
2020-12-10 15:11:06 +03:00
struct fio * fio = fruit_get_complete_fio ( handle , fsp ) ;
2016-12-08 22:38:17 +03:00
ssize_t nread ;
2014-06-23 18:59:45 +04:00
2018-04-10 22:05:09 +03:00
if ( fio = = NULL ) {
errno = EINVAL ;
return - 1 ;
}
2016-12-08 22:38:17 +03:00
switch ( fio - > config - > rsrc ) {
case FRUIT_RSRC_STREAM :
nread = fruit_pread_rsrc_stream ( handle , fsp , data , n , offset ) ;
break ;
2016-11-16 13:01:45 +03:00
2016-12-08 22:38:17 +03:00
case FRUIT_RSRC_ADFILE :
nread = fruit_pread_rsrc_adouble ( handle , fsp , data , n , offset ) ;
break ;
2016-11-16 13:01:45 +03:00
2016-12-08 22:38:17 +03:00
case FRUIT_RSRC_XATTR :
nread = fruit_pread_rsrc_xattr ( handle , fsp , data , n , offset ) ;
break ;
2015-06-25 16:42:04 +03:00
2016-12-08 22:38:17 +03:00
default :
DBG_ERR ( " Unexpected rsrc config [%d] \n " , fio - > config - > rsrc ) ;
return - 1 ;
2014-06-23 18:59:45 +04:00
}
2016-12-08 22:38:17 +03:00
return nread ;
}
static ssize_t fruit_pread ( vfs_handle_struct * handle ,
files_struct * fsp , void * data ,
size_t n , off_t offset )
{
2020-12-10 15:11:06 +03:00
struct fio * fio = fruit_get_complete_fio ( handle , fsp ) ;
2016-12-08 22:38:17 +03:00
ssize_t nread ;
2017-04-29 13:01:41 +03:00
DBG_DEBUG ( " Path [%s] offset=% " PRIdMAX " , size=%zd \n " ,
fsp_str_dbg ( fsp ) , ( intmax_t ) offset , n ) ;
2016-12-08 22:38:17 +03:00
if ( fio = = NULL ) {
return SMB_VFS_NEXT_PREAD ( handle , fsp , data , n , offset ) ;
2014-06-23 18:59:45 +04:00
}
2016-12-08 22:38:17 +03:00
if ( fio - > type = = ADOUBLE_META ) {
nread = fruit_pread_meta ( handle , fsp , data , n , offset ) ;
} else {
nread = fruit_pread_rsrc ( handle , fsp , data , n , offset ) ;
}
DBG_DEBUG ( " Path [%s] nread [%zd] \n " , fsp_str_dbg ( fsp ) , nread ) ;
return nread ;
2014-06-23 18:59:45 +04:00
}
2017-05-12 15:40:03 +03:00
static bool fruit_must_handle_aio_stream ( struct fio * fio )
{
if ( fio = = NULL ) {
return false ;
} ;
2018-10-17 17:51:34 +03:00
if ( fio - > type = = ADOUBLE_META ) {
2017-05-12 15:40:03 +03:00
return true ;
}
if ( ( fio - > type = = ADOUBLE_RSRC ) & &
( fio - > config - > rsrc = = FRUIT_RSRC_ADFILE ) )
{
return true ;
}
return false ;
}
struct fruit_pread_state {
ssize_t nread ;
struct vfs_aio_state vfs_aio_state ;
} ;
static void fruit_pread_done ( struct tevent_req * subreq ) ;
static struct tevent_req * fruit_pread_send (
struct vfs_handle_struct * handle ,
TALLOC_CTX * mem_ctx ,
struct tevent_context * ev ,
struct files_struct * fsp ,
void * data ,
size_t n , off_t offset )
{
struct tevent_req * req = NULL ;
struct tevent_req * subreq = NULL ;
struct fruit_pread_state * state = NULL ;
2020-12-10 15:11:06 +03:00
struct fio * fio = fruit_get_complete_fio ( handle , fsp ) ;
2017-05-12 15:40:03 +03:00
req = tevent_req_create ( mem_ctx , & state ,
struct fruit_pread_state ) ;
if ( req = = NULL ) {
return NULL ;
}
if ( fruit_must_handle_aio_stream ( fio ) ) {
state - > nread = SMB_VFS_PREAD ( fsp , data , n , offset ) ;
if ( state - > nread ! = n ) {
if ( state - > nread ! = - 1 ) {
errno = EIO ;
}
tevent_req_error ( req , errno ) ;
return tevent_req_post ( req , ev ) ;
}
tevent_req_done ( req ) ;
return tevent_req_post ( req , ev ) ;
}
subreq = SMB_VFS_NEXT_PREAD_SEND ( state , ev , handle , fsp ,
data , n , offset ) ;
if ( tevent_req_nomem ( req , subreq ) ) {
return tevent_req_post ( req , ev ) ;
}
tevent_req_set_callback ( subreq , fruit_pread_done , req ) ;
return req ;
}
static void fruit_pread_done ( struct tevent_req * subreq )
{
struct tevent_req * req = tevent_req_callback_data (
subreq , struct tevent_req ) ;
struct fruit_pread_state * state = tevent_req_data (
req , struct fruit_pread_state ) ;
state - > nread = SMB_VFS_PREAD_RECV ( subreq , & state - > vfs_aio_state ) ;
TALLOC_FREE ( subreq ) ;
if ( tevent_req_error ( req , state - > vfs_aio_state . error ) ) {
return ;
}
tevent_req_done ( req ) ;
}
static ssize_t fruit_pread_recv ( struct tevent_req * req ,
struct vfs_aio_state * vfs_aio_state )
{
struct fruit_pread_state * state = tevent_req_data (
req , struct fruit_pread_state ) ;
2022-10-06 15:31:08 +03:00
ssize_t retval = - 1 ;
2017-05-12 15:40:03 +03:00
if ( tevent_req_is_unix_error ( req , & vfs_aio_state - > error ) ) {
2022-10-06 15:31:08 +03:00
tevent_req_received ( req ) ;
2017-05-12 15:40:03 +03:00
return - 1 ;
}
* vfs_aio_state = state - > vfs_aio_state ;
2022-10-06 15:31:08 +03:00
retval = state - > nread ;
tevent_req_received ( req ) ;
return retval ;
2017-05-12 15:40:03 +03:00
}
2016-12-08 22:38:17 +03:00
static ssize_t fruit_pwrite_meta_stream ( vfs_handle_struct * handle ,
2023-07-04 18:46:40 +03:00
files_struct * fsp , const void * indata ,
2016-12-08 22:38:17 +03:00
size_t n , off_t offset )
2014-06-23 18:59:45 +04:00
{
2020-12-10 15:11:06 +03:00
struct fio * fio = fruit_get_complete_fio ( handle , fsp ) ;
2023-07-04 18:46:40 +03:00
const void * data = indata ;
char afpinfo_buf [ AFP_INFO_SIZE ] ;
2014-06-23 18:59:45 +04:00
AfpInfo * ai = NULL ;
2017-12-07 00:09:52 +03:00
size_t nwritten ;
2018-08-22 16:21:08 +03:00
int ret ;
2017-12-07 00:09:52 +03:00
bool ok ;
2014-06-23 18:59:45 +04:00
2018-08-22 16:21:08 +03:00
DBG_DEBUG ( " Path [%s] offset=% " PRIdMAX " , size=%zd \n " ,
fsp_str_dbg ( fsp ) , ( intmax_t ) offset , n ) ;
if ( fio = = NULL ) {
return - 1 ;
}
if ( fio - > fake_fd ) {
2022-06-03 16:53:29 +03:00
struct vfs_open_how how = {
. flags = fio - > flags , . mode = fio - > mode ,
} ;
2020-12-11 14:59:28 +03:00
int fd = fsp_get_pathref_fd ( fsp ) ;
2018-08-22 16:21:08 +03:00
2020-12-11 14:59:28 +03:00
ret = vfs_fake_fd_close ( fd ) ;
2020-09-26 22:46:51 +03:00
fsp_set_fd ( fsp , - 1 ) ;
2018-08-22 16:21:08 +03:00
if ( ret ! = 0 ) {
DBG_ERR ( " Close [%s] failed: %s \n " ,
fsp_str_dbg ( fsp ) , strerror ( errno ) ) ;
return - 1 ;
}
2020-05-21 00:05:07 +03:00
fd = SMB_VFS_NEXT_OPENAT ( handle ,
2022-06-14 17:33:57 +03:00
NULL , /* opening a stream */
2020-05-21 00:05:07 +03:00
fsp - > fsp_name ,
fsp ,
2022-06-03 16:53:29 +03:00
& how ) ;
2018-08-22 16:21:08 +03:00
if ( fd = = - 1 ) {
DBG_ERR ( " On-demand create [%s] in write failed: %s \n " ,
fsp_str_dbg ( fsp ) , strerror ( errno ) ) ;
return - 1 ;
}
2020-09-26 22:46:51 +03:00
fsp_set_fd ( fsp , fd ) ;
2018-08-22 16:21:08 +03:00
fio - > fake_fd = false ;
}
2023-07-04 18:46:40 +03:00
ai = afpinfo_unpack ( talloc_tos ( ) , data , fio - > config - > validate_afpinfo ) ;
2016-12-08 22:38:17 +03:00
if ( ai = = NULL ) {
return - 1 ;
}
2014-06-23 18:59:45 +04:00
2018-10-21 00:40:14 +03:00
if ( ai_empty_finderinfo ( ai ) ) {
2018-11-06 14:34:17 +03:00
/*
* Writing an all 0 blob to the metadata stream results in the
* stream being removed on a macOS server . This ensures we
* behave the same and it verified by the " delete AFP_AfpInfo by
* writing all 0 " test.
*/
2018-10-21 00:40:14 +03:00
ret = SMB_VFS_NEXT_FTRUNCATE ( handle , fsp , 0 ) ;
if ( ret ! = 0 ) {
DBG_ERR ( " SMB_VFS_NEXT_FTRUNCATE on [%s] failed \n " ,
fsp_str_dbg ( fsp ) ) ;
return - 1 ;
}
2014-06-23 18:59:45 +04:00
2018-10-21 00:40:14 +03:00
ok = set_delete_on_close (
2017-12-07 00:09:52 +03:00
fsp ,
true ,
handle - > conn - > session_info - > security_token ,
handle - > conn - > session_info - > unix_token ) ;
2018-10-21 00:40:14 +03:00
if ( ! ok ) {
DBG_ERR ( " set_delete_on_close on [%s] failed \n " ,
fsp_str_dbg ( fsp ) ) ;
return - 1 ;
}
return n ;
}
2023-07-04 18:46:40 +03:00
if ( ! fio - > config - > validate_afpinfo ) {
/*
* Ensure the buffer contains a valid header , so marshall
* the data from the afpinfo struck back into a buffer
* and write that instead of the possibly malformed data
* we got from the client .
*/
nwritten = afpinfo_pack ( ai , afpinfo_buf ) ;
if ( nwritten ! = AFP_INFO_SIZE ) {
errno = EINVAL ;
return - 1 ;
}
data = afpinfo_buf ;
}
2018-10-21 00:40:14 +03:00
nwritten = SMB_VFS_NEXT_PWRITE ( handle , fsp , data , n , offset ) ;
if ( nwritten ! = n ) {
2017-12-07 00:09:52 +03:00
return - 1 ;
}
return n ;
2016-12-08 22:38:17 +03:00
}
2014-06-23 18:59:45 +04:00
2016-12-08 22:38:17 +03:00
static ssize_t fruit_pwrite_meta_netatalk ( vfs_handle_struct * handle ,
files_struct * fsp , const void * data ,
size_t n , off_t offset )
{
2023-07-04 18:46:40 +03:00
struct fruit_config_data * config = NULL ;
2016-12-08 22:38:17 +03:00
struct adouble * ad = NULL ;
AfpInfo * ai = NULL ;
char * p = NULL ;
int ret ;
2017-12-07 00:09:52 +03:00
bool ok ;
2016-11-15 23:32:25 +03:00
2023-07-04 18:46:40 +03:00
SMB_VFS_HANDLE_GET_DATA ( handle , config ,
struct fruit_config_data , return - 1 ) ;
ai = afpinfo_unpack ( talloc_tos ( ) , data , config - > validate_afpinfo ) ;
2016-12-08 22:38:17 +03:00
if ( ai = = NULL ) {
return - 1 ;
}
2016-11-15 23:32:25 +03:00
2016-12-08 22:38:17 +03:00
ad = ad_fget ( talloc_tos ( ) , handle , fsp , ADOUBLE_META ) ;
2016-11-15 23:32:25 +03:00
if ( ad = = NULL ) {
2019-05-22 22:15:22 +03:00
ad = ad_init ( talloc_tos ( ) , ADOUBLE_META ) ;
2016-12-08 22:38:17 +03:00
if ( ad = = NULL ) {
return - 1 ;
2016-11-15 23:32:25 +03:00
}
2016-12-08 22:38:17 +03:00
}
p = ad_get_entry ( ad , ADEID_FINDERI ) ;
if ( p = = NULL ) {
DBG_ERR ( " No ADEID_FINDERI for [%s] \n " , fsp_str_dbg ( fsp ) ) ;
TALLOC_FREE ( ad ) ;
return - 1 ;
2016-11-15 23:32:25 +03:00
}
2016-12-08 22:38:17 +03:00
memcpy ( p , & ai - > afpi_FinderInfo [ 0 ] , ADEDLEN_FINDERI ) ;
2016-11-15 23:32:25 +03:00
2019-05-17 11:41:29 +03:00
ret = ad_fset ( handle , ad , fsp ) ;
2016-12-08 22:38:17 +03:00
if ( ret ! = 0 ) {
DBG_ERR ( " ad_pwrite [%s] failed \n " , fsp_str_dbg ( fsp ) ) ;
TALLOC_FREE ( ad ) ;
return - 1 ;
}
2016-11-16 13:01:45 +03:00
2016-12-08 22:38:17 +03:00
TALLOC_FREE ( ad ) ;
2017-12-07 00:09:52 +03:00
if ( ! ai_empty_finderinfo ( ai ) ) {
return n ;
}
2018-11-06 14:34:17 +03:00
/*
* Writing an all 0 blob to the metadata stream results in the stream
* being removed on a macOS server . This ensures we behave the same and
* it verified by the " delete AFP_AfpInfo by writing all 0 " test .
*/
2017-12-07 00:09:52 +03:00
ok = set_delete_on_close (
fsp ,
true ,
handle - > conn - > session_info - > security_token ,
handle - > conn - > session_info - > unix_token ) ;
if ( ! ok ) {
DBG_ERR ( " set_delete_on_close on [%s] failed \n " ,
fsp_str_dbg ( fsp ) ) ;
return - 1 ;
}
2016-12-08 22:38:17 +03:00
return n ;
}
2014-06-23 18:59:45 +04:00
2016-12-08 22:38:17 +03:00
static ssize_t fruit_pwrite_meta ( vfs_handle_struct * handle ,
files_struct * fsp , const void * data ,
size_t n , off_t offset )
{
2020-12-10 15:11:06 +03:00
struct fio * fio = fruit_get_complete_fio ( handle , fsp ) ;
2016-12-08 22:38:17 +03:00
ssize_t nwritten ;
2018-11-06 15:24:14 +03:00
uint8_t buf [ AFP_INFO_SIZE ] ;
size_t to_write ;
size_t to_copy ;
int cmp ;
2014-06-23 18:59:45 +04:00
2018-11-06 15:24:14 +03:00
if ( fio = = NULL ) {
2023-08-07 07:37:51 +03:00
DBG_ERR ( " Failed to fetch fsp extension \n " ) ;
2016-12-08 22:38:17 +03:00
return - 1 ;
2014-06-23 18:59:45 +04:00
}
2018-11-06 15:24:14 +03:00
if ( n < 3 ) {
errno = EINVAL ;
2018-05-25 12:39:50 +03:00
return - 1 ;
}
2018-11-06 15:24:14 +03:00
if ( offset ! = 0 & & n < 60 ) {
errno = EINVAL ;
return - 1 ;
}
2023-07-04 18:46:40 +03:00
if ( fio - > config - > validate_afpinfo ) {
cmp = memcmp ( data , " AFP " , 3 ) ;
if ( cmp ! = 0 ) {
errno = EINVAL ;
return - 1 ;
}
2018-11-06 15:24:14 +03:00
}
if ( n < = AFP_OFF_FinderInfo ) {
/*
* Nothing to do here really , just return
*/
return n ;
}
offset = 0 ;
to_copy = n ;
if ( to_copy > AFP_INFO_SIZE ) {
to_copy = AFP_INFO_SIZE ;
}
memcpy ( buf , data , to_copy ) ;
to_write = n ;
if ( to_write ! = AFP_INFO_SIZE ) {
to_write = AFP_INFO_SIZE ;
}
2016-12-08 22:38:17 +03:00
switch ( fio - > config - > meta ) {
case FRUIT_META_STREAM :
2018-11-06 15:24:14 +03:00
nwritten = fruit_pwrite_meta_stream ( handle ,
fsp ,
buf ,
to_write ,
offset ) ;
2016-12-08 22:38:17 +03:00
break ;
case FRUIT_META_NETATALK :
2018-11-06 15:24:14 +03:00
nwritten = fruit_pwrite_meta_netatalk ( handle ,
fsp ,
buf ,
to_write ,
offset ) ;
2016-12-08 22:38:17 +03:00
break ;
default :
DBG_ERR ( " Unexpected meta config [%d] \n " , fio - > config - > meta ) ;
return - 1 ;
}
2018-11-06 15:24:14 +03:00
if ( nwritten ! = to_write ) {
return - 1 ;
}
/*
* Return the requested amount , verified against macOS SMB server
*/
return n ;
2016-12-08 22:38:17 +03:00
}
static ssize_t fruit_pwrite_rsrc_stream ( vfs_handle_struct * handle ,
files_struct * fsp , const void * data ,
size_t n , off_t offset )
{
return SMB_VFS_NEXT_PWRITE ( handle , fsp , data , n , offset ) ;
}
static ssize_t fruit_pwrite_rsrc_xattr ( vfs_handle_struct * handle ,
files_struct * fsp , const void * data ,
size_t n , off_t offset )
{
return SMB_VFS_NEXT_PWRITE ( handle , fsp , data , n , offset ) ;
}
static ssize_t fruit_pwrite_rsrc_adouble ( vfs_handle_struct * handle ,
files_struct * fsp , const void * data ,
size_t n , off_t offset )
{
2020-12-08 18:29:10 +03:00
struct fio * fio = fruit_get_complete_fio ( handle , fsp ) ;
2016-12-08 22:38:17 +03:00
struct adouble * ad = NULL ;
ssize_t nwritten ;
int ret ;
2022-01-10 15:26:25 +03:00
if ( fio = = NULL | | fio - > ad_fsp = = NULL ) {
DBG_ERR ( " fio/ad_fsp=NULL for [%s] \n " , fsp_str_dbg ( fsp ) ) ;
2020-12-08 18:29:10 +03:00
errno = EBADF ;
return - 1 ;
}
ad = ad_fget ( talloc_tos ( ) , handle , fio - > ad_fsp , ADOUBLE_RSRC ) ;
2016-12-08 22:38:17 +03:00
if ( ad = = NULL ) {
2020-12-08 18:29:10 +03:00
DBG_ERR ( " ad_fget [%s] failed [%s] \n " ,
fsp_str_dbg ( fio - > ad_fsp ) , strerror ( errno ) ) ;
2014-06-23 18:59:45 +04:00
return - 1 ;
}
2016-12-08 22:38:17 +03:00
2020-12-08 18:29:10 +03:00
nwritten = SMB_VFS_NEXT_PWRITE ( handle , fio - > ad_fsp , data , n ,
2016-12-08 22:38:17 +03:00
offset + ad_getentryoff ( ad , ADEID_RFORK ) ) ;
if ( nwritten ! = n ) {
DBG_ERR ( " Short write on [%s] [%zd/%zd] \n " ,
2020-12-08 18:29:10 +03:00
fsp_str_dbg ( fio - > ad_fsp ) , nwritten , n ) ;
2016-12-08 22:38:17 +03:00
TALLOC_FREE ( ad ) ;
return - 1 ;
}
if ( ( n + offset ) > ad_getentrylen ( ad , ADEID_RFORK ) ) {
ad_setentrylen ( ad , ADEID_RFORK , n + offset ) ;
2020-12-08 18:29:10 +03:00
ret = ad_fset ( handle , ad , fio - > ad_fsp ) ;
2016-12-08 22:38:17 +03:00
if ( ret ! = 0 ) {
2020-12-08 18:29:10 +03:00
DBG_ERR ( " ad_pwrite [%s] failed \n " , fsp_str_dbg ( fio - > ad_fsp ) ) ;
2016-12-08 22:38:17 +03:00
TALLOC_FREE ( ad ) ;
return - 1 ;
}
}
TALLOC_FREE ( ad ) ;
2014-06-23 18:59:45 +04:00
return n ;
}
2016-12-08 22:38:17 +03:00
static ssize_t fruit_pwrite_rsrc ( vfs_handle_struct * handle ,
files_struct * fsp , const void * data ,
size_t n , off_t offset )
{
2020-12-10 15:11:06 +03:00
struct fio * fio = fruit_get_complete_fio ( handle , fsp ) ;
2016-12-08 22:38:17 +03:00
ssize_t nwritten ;
2018-05-25 12:32:38 +03:00
if ( fio = = NULL ) {
2023-08-07 07:37:51 +03:00
DBG_ERR ( " Failed to fetch fsp extension \n " ) ;
2018-05-25 12:32:38 +03:00
return - 1 ;
}
2016-12-08 22:38:17 +03:00
switch ( fio - > config - > rsrc ) {
case FRUIT_RSRC_STREAM :
nwritten = fruit_pwrite_rsrc_stream ( handle , fsp , data , n , offset ) ;
break ;
case FRUIT_RSRC_ADFILE :
nwritten = fruit_pwrite_rsrc_adouble ( handle , fsp , data , n , offset ) ;
break ;
case FRUIT_RSRC_XATTR :
nwritten = fruit_pwrite_rsrc_xattr ( handle , fsp , data , n , offset ) ;
break ;
default :
DBG_ERR ( " Unexpected rsrc config [%d] \n " , fio - > config - > rsrc ) ;
return - 1 ;
}
return nwritten ;
}
static ssize_t fruit_pwrite ( vfs_handle_struct * handle ,
files_struct * fsp , const void * data ,
size_t n , off_t offset )
{
2020-12-10 15:11:06 +03:00
struct fio * fio = fruit_get_complete_fio ( handle , fsp ) ;
2016-12-08 22:38:17 +03:00
ssize_t nwritten ;
2017-04-29 13:01:41 +03:00
DBG_DEBUG ( " Path [%s] offset=% " PRIdMAX " , size=%zd \n " ,
fsp_str_dbg ( fsp ) , ( intmax_t ) offset , n ) ;
2016-12-08 22:38:17 +03:00
if ( fio = = NULL ) {
return SMB_VFS_NEXT_PWRITE ( handle , fsp , data , n , offset ) ;
}
if ( fio - > type = = ADOUBLE_META ) {
nwritten = fruit_pwrite_meta ( handle , fsp , data , n , offset ) ;
} else {
nwritten = fruit_pwrite_rsrc ( handle , fsp , data , n , offset ) ;
}
DBG_DEBUG ( " Path [%s] nwritten=%zd \n " , fsp_str_dbg ( fsp ) , nwritten ) ;
return nwritten ;
}
2017-05-12 15:40:03 +03:00
struct fruit_pwrite_state {
ssize_t nwritten ;
struct vfs_aio_state vfs_aio_state ;
} ;
static void fruit_pwrite_done ( struct tevent_req * subreq ) ;
static struct tevent_req * fruit_pwrite_send (
struct vfs_handle_struct * handle ,
TALLOC_CTX * mem_ctx ,
struct tevent_context * ev ,
struct files_struct * fsp ,
const void * data ,
size_t n , off_t offset )
{
struct tevent_req * req = NULL ;
struct tevent_req * subreq = NULL ;
struct fruit_pwrite_state * state = NULL ;
2020-12-10 15:11:06 +03:00
struct fio * fio = fruit_get_complete_fio ( handle , fsp ) ;
2017-05-12 15:40:03 +03:00
req = tevent_req_create ( mem_ctx , & state ,
struct fruit_pwrite_state ) ;
if ( req = = NULL ) {
return NULL ;
}
if ( fruit_must_handle_aio_stream ( fio ) ) {
state - > nwritten = SMB_VFS_PWRITE ( fsp , data , n , offset ) ;
if ( state - > nwritten ! = n ) {
if ( state - > nwritten ! = - 1 ) {
errno = EIO ;
}
tevent_req_error ( req , errno ) ;
return tevent_req_post ( req , ev ) ;
}
tevent_req_done ( req ) ;
return tevent_req_post ( req , ev ) ;
}
subreq = SMB_VFS_NEXT_PWRITE_SEND ( state , ev , handle , fsp ,
data , n , offset ) ;
if ( tevent_req_nomem ( req , subreq ) ) {
return tevent_req_post ( req , ev ) ;
}
tevent_req_set_callback ( subreq , fruit_pwrite_done , req ) ;
return req ;
}
static void fruit_pwrite_done ( struct tevent_req * subreq )
{
struct tevent_req * req = tevent_req_callback_data (
subreq , struct tevent_req ) ;
struct fruit_pwrite_state * state = tevent_req_data (
req , struct fruit_pwrite_state ) ;
state - > nwritten = SMB_VFS_PWRITE_RECV ( subreq , & state - > vfs_aio_state ) ;
TALLOC_FREE ( subreq ) ;
if ( tevent_req_error ( req , state - > vfs_aio_state . error ) ) {
return ;
}
tevent_req_done ( req ) ;
}
static ssize_t fruit_pwrite_recv ( struct tevent_req * req ,
struct vfs_aio_state * vfs_aio_state )
{
struct fruit_pwrite_state * state = tevent_req_data (
req , struct fruit_pwrite_state ) ;
2022-10-06 15:31:08 +03:00
ssize_t retval = - 1 ;
2017-05-12 15:40:03 +03:00
if ( tevent_req_is_unix_error ( req , & vfs_aio_state - > error ) ) {
2022-10-06 15:31:08 +03:00
tevent_req_received ( req ) ;
2017-05-12 15:40:03 +03:00
return - 1 ;
}
* vfs_aio_state = state - > vfs_aio_state ;
2022-10-06 15:31:08 +03:00
retval = state - > nwritten ;
tevent_req_received ( req ) ;
return retval ;
2017-05-12 15:40:03 +03:00
}
2022-09-20 23:25:22 +03:00
struct fruit_fsync_state {
int ret ;
struct vfs_aio_state vfs_aio_state ;
} ;
static void fruit_fsync_done ( struct tevent_req * subreq ) ;
static struct tevent_req * fruit_fsync_send (
struct vfs_handle_struct * handle ,
TALLOC_CTX * mem_ctx ,
struct tevent_context * ev ,
struct files_struct * fsp )
{
struct tevent_req * req = NULL ;
struct tevent_req * subreq = NULL ;
struct fruit_fsync_state * state = NULL ;
struct fio * fio = fruit_get_complete_fio ( handle , fsp ) ;
req = tevent_req_create ( mem_ctx , & state ,
struct fruit_fsync_state ) ;
if ( req = = NULL ) {
return NULL ;
}
if ( fruit_must_handle_aio_stream ( fio ) ) {
struct adouble * ad = NULL ;
if ( fio - > type = = ADOUBLE_META ) {
/*
* We must never pass a fake_fd
* to lower level fsync calls .
* Everything is already done
* synchronously , so just return
* true .
*/
SMB_ASSERT ( fio - > fake_fd ) ;
tevent_req_done ( req ) ;
return tevent_req_post ( req , ev ) ;
}
/*
* We know the following must be true ,
* as it ' s the condition for fruit_must_handle_aio_stream ( )
* to return true if fio - > type = = ADOUBLE_RSRC .
*/
SMB_ASSERT ( fio - > config - > rsrc = = FRUIT_RSRC_ADFILE ) ;
if ( fio - > ad_fsp = = NULL ) {
tevent_req_error ( req , EBADF ) ;
return tevent_req_post ( req , ev ) ;
}
ad = ad_fget ( talloc_tos ( ) , handle , fio - > ad_fsp , ADOUBLE_RSRC ) ;
if ( ad = = NULL ) {
tevent_req_error ( req , ENOMEM ) ;
return tevent_req_post ( req , ev ) ;
}
fsp = fio - > ad_fsp ;
}
subreq = SMB_VFS_NEXT_FSYNC_SEND ( state , ev , handle , fsp ) ;
if ( tevent_req_nomem ( req , subreq ) ) {
return tevent_req_post ( req , ev ) ;
}
tevent_req_set_callback ( subreq , fruit_fsync_done , req ) ;
return req ;
}
static void fruit_fsync_done ( struct tevent_req * subreq )
{
struct tevent_req * req = tevent_req_callback_data (
subreq , struct tevent_req ) ;
struct fruit_fsync_state * state = tevent_req_data (
req , struct fruit_fsync_state ) ;
state - > ret = SMB_VFS_FSYNC_RECV ( subreq , & state - > vfs_aio_state ) ;
TALLOC_FREE ( subreq ) ;
if ( state - > ret ! = 0 ) {
tevent_req_error ( req , errno ) ;
return ;
}
tevent_req_done ( req ) ;
}
static int fruit_fsync_recv ( struct tevent_req * req ,
struct vfs_aio_state * vfs_aio_state )
{
struct fruit_fsync_state * state = tevent_req_data (
req , struct fruit_fsync_state ) ;
int retval = - 1 ;
if ( tevent_req_is_unix_error ( req , & vfs_aio_state - > error ) ) {
tevent_req_received ( req ) ;
return - 1 ;
}
* vfs_aio_state = state - > vfs_aio_state ;
retval = state - > ret ;
tevent_req_received ( req ) ;
return retval ;
}
2014-06-23 18:59:45 +04:00
/**
* Helper to stat / lstat the base file of an smb_fname .
*/
static int fruit_stat_base ( vfs_handle_struct * handle ,
struct smb_filename * smb_fname ,
bool follow_links )
{
char * tmp_stream_name ;
int rc ;
tmp_stream_name = smb_fname - > stream_name ;
smb_fname - > stream_name = NULL ;
if ( follow_links ) {
rc = SMB_VFS_NEXT_STAT ( handle , smb_fname ) ;
} else {
rc = SMB_VFS_NEXT_LSTAT ( handle , smb_fname ) ;
}
smb_fname - > stream_name = tmp_stream_name ;
2018-10-17 20:07:11 +03:00
DBG_DEBUG ( " fruit_stat_base [%s] dev [%ju] ino [%ju] \n " ,
smb_fname - > base_name ,
( uintmax_t ) smb_fname - > st . st_ex_dev ,
( uintmax_t ) smb_fname - > st . st_ex_ino ) ;
2014-06-23 18:59:45 +04:00
return rc ;
}
2016-12-02 13:12:18 +03:00
static int fruit_stat_meta_stream ( vfs_handle_struct * handle ,
struct smb_filename * smb_fname ,
bool follow_links )
{
int ret ;
2018-08-22 17:49:23 +03:00
ino_t ino ;
ret = fruit_stat_base ( handle , smb_fname , false ) ;
if ( ret ! = 0 ) {
return - 1 ;
}
2019-06-18 17:58:29 +03:00
ino = hash_inode ( & smb_fname - > st , smb_fname - > stream_name ) ;
2016-12-02 13:12:18 +03:00
if ( follow_links ) {
ret = SMB_VFS_NEXT_STAT ( handle , smb_fname ) ;
} else {
ret = SMB_VFS_NEXT_LSTAT ( handle , smb_fname ) ;
}
2018-08-22 17:49:23 +03:00
smb_fname - > st . st_ex_ino = ino ;
2016-12-02 13:12:18 +03:00
return ret ;
}
2016-12-02 13:05:50 +03:00
static int fruit_stat_meta_netatalk ( vfs_handle_struct * handle ,
struct smb_filename * smb_fname ,
bool follow_links )
2014-06-23 18:59:45 +04:00
{
2015-12-17 22:05:04 +03:00
struct adouble * ad = NULL ;
2021-07-02 23:37:56 +03:00
/* Populate the stat struct with info from the base file. */
if ( fruit_stat_base ( handle , smb_fname , follow_links ) = = - 1 ) {
return - 1 ;
}
2021-07-02 23:53:02 +03:00
ad = ad_get_meta_fsp ( talloc_tos ( ) , handle , smb_fname ) ;
2015-12-17 22:05:04 +03:00
if ( ad = = NULL ) {
DBG_INFO ( " fruit_stat_meta %s: %s \n " ,
smb_fname_str_dbg ( smb_fname ) , strerror ( errno ) ) ;
errno = ENOENT ;
return - 1 ;
}
TALLOC_FREE ( ad ) ;
2014-06-23 18:59:45 +04:00
smb_fname - > st . st_ex_size = AFP_INFO_SIZE ;
2019-06-18 17:58:29 +03:00
smb_fname - > st . st_ex_ino = hash_inode ( & smb_fname - > st ,
2014-06-23 18:59:45 +04:00
smb_fname - > stream_name ) ;
return 0 ;
}
2016-12-02 13:05:50 +03:00
static int fruit_stat_meta ( vfs_handle_struct * handle ,
struct smb_filename * smb_fname ,
bool follow_links )
{
struct fruit_config_data * config = NULL ;
int ret ;
SMB_VFS_HANDLE_GET_DATA ( handle , config ,
struct fruit_config_data , return - 1 ) ;
switch ( config - > meta ) {
case FRUIT_META_STREAM :
2016-12-02 13:12:18 +03:00
ret = fruit_stat_meta_stream ( handle , smb_fname , follow_links ) ;
break ;
2016-12-02 13:05:50 +03:00
case FRUIT_META_NETATALK :
ret = fruit_stat_meta_netatalk ( handle , smb_fname , follow_links ) ;
break ;
default :
DBG_ERR ( " Unexpected meta config [%d] \n " , config - > meta ) ;
return - 1 ;
}
return ret ;
}
2016-12-02 13:26:22 +03:00
static int fruit_stat_rsrc_netatalk ( vfs_handle_struct * handle ,
struct smb_filename * smb_fname ,
bool follow_links )
2014-06-23 18:59:45 +04:00
{
struct adouble * ad = NULL ;
2016-12-02 13:26:22 +03:00
int ret ;
2014-06-23 18:59:45 +04:00
2017-05-25 21:38:26 +03:00
ad = ad_get ( talloc_tos ( ) , handle , smb_fname , ADOUBLE_RSRC ) ;
2014-06-23 18:59:45 +04:00
if ( ad = = NULL ) {
errno = ENOENT ;
return - 1 ;
}
/* Populate the stat struct with info from the base file. */
2016-12-02 13:26:22 +03:00
ret = fruit_stat_base ( handle , smb_fname , follow_links ) ;
if ( ret ! = 0 ) {
2014-06-23 18:59:45 +04:00
TALLOC_FREE ( ad ) ;
return - 1 ;
}
smb_fname - > st . st_ex_size = ad_getentrylen ( ad , ADEID_RFORK ) ;
2019-06-18 17:58:29 +03:00
smb_fname - > st . st_ex_ino = hash_inode ( & smb_fname - > st ,
2014-06-23 18:59:45 +04:00
smb_fname - > stream_name ) ;
TALLOC_FREE ( ad ) ;
return 0 ;
}
2016-12-02 13:30:06 +03:00
static int fruit_stat_rsrc_stream ( vfs_handle_struct * handle ,
struct smb_filename * smb_fname ,
bool follow_links )
{
int ret ;
if ( follow_links ) {
ret = SMB_VFS_NEXT_STAT ( handle , smb_fname ) ;
} else {
ret = SMB_VFS_NEXT_LSTAT ( handle , smb_fname ) ;
}
return ret ;
}
2016-12-02 13:44:53 +03:00
static int fruit_stat_rsrc_xattr ( vfs_handle_struct * handle ,
struct smb_filename * smb_fname ,
bool follow_links )
{
# ifdef HAVE_ATTROPEN
int ret ;
int fd = - 1 ;
/* Populate the stat struct with info from the base file. */
ret = fruit_stat_base ( handle , smb_fname , follow_links ) ;
if ( ret ! = 0 ) {
return - 1 ;
}
fd = attropen ( smb_fname - > base_name ,
AFPRESOURCE_EA_NETATALK ,
O_RDONLY ) ;
if ( fd = = - 1 ) {
return 0 ;
}
ret = sys_fstat ( fd , & smb_fname - > st , false ) ;
if ( ret ! = 0 ) {
close ( fd ) ;
DBG_ERR ( " fstat [%s:%s] failed \n " , smb_fname - > base_name ,
AFPRESOURCE_EA_NETATALK ) ;
return - 1 ;
}
close ( fd ) ;
fd = - 1 ;
2019-06-18 17:58:29 +03:00
smb_fname - > st . st_ex_ino = hash_inode ( & smb_fname - > st ,
smb_fname - > stream_name ) ;
2016-12-02 13:44:53 +03:00
return ret ;
# else
errno = ENOSYS ;
return - 1 ;
# endif
}
2016-12-02 13:26:22 +03:00
static int fruit_stat_rsrc ( vfs_handle_struct * handle ,
struct smb_filename * smb_fname ,
bool follow_links )
{
struct fruit_config_data * config = NULL ;
int ret ;
DBG_DEBUG ( " Path [%s] \n " , smb_fname_str_dbg ( smb_fname ) ) ;
SMB_VFS_HANDLE_GET_DATA ( handle , config ,
struct fruit_config_data , return - 1 ) ;
switch ( config - > rsrc ) {
case FRUIT_RSRC_STREAM :
2016-12-02 13:30:06 +03:00
ret = fruit_stat_rsrc_stream ( handle , smb_fname , follow_links ) ;
break ;
2016-12-02 13:26:22 +03:00
case FRUIT_RSRC_XATTR :
2016-12-02 13:44:53 +03:00
ret = fruit_stat_rsrc_xattr ( handle , smb_fname , follow_links ) ;
break ;
2016-12-02 13:26:22 +03:00
case FRUIT_RSRC_ADFILE :
ret = fruit_stat_rsrc_netatalk ( handle , smb_fname , follow_links ) ;
break ;
default :
DBG_ERR ( " Unexpected rsrc config [%d] \n " , config - > rsrc ) ;
return - 1 ;
}
return ret ;
}
2014-06-23 18:59:45 +04:00
static int fruit_stat ( vfs_handle_struct * handle ,
struct smb_filename * smb_fname )
{
int rc = - 1 ;
DEBUG ( 10 , ( " fruit_stat called for %s \n " ,
smb_fname_str_dbg ( smb_fname ) ) ) ;
2019-09-26 20:31:51 +03:00
if ( ! is_named_stream ( smb_fname ) ) {
2014-06-23 18:59:45 +04:00
rc = SMB_VFS_NEXT_STAT ( handle , smb_fname ) ;
if ( rc = = 0 ) {
update_btime ( handle , smb_fname ) ;
}
return rc ;
}
/*
* Note if lp_posix_paths ( ) is true , we can never
* get here as is_ntfs_stream_smb_fname ( ) is
* always false . So we never need worry about
* not following links here .
*/
2019-07-01 16:53:36 +03:00
if ( is_afpinfo_stream ( smb_fname - > stream_name ) ) {
2014-06-23 18:59:45 +04:00
rc = fruit_stat_meta ( handle , smb_fname , true ) ;
2019-07-01 16:53:36 +03:00
} else if ( is_afpresource_stream ( smb_fname - > stream_name ) ) {
2014-06-23 18:59:45 +04:00
rc = fruit_stat_rsrc ( handle , smb_fname , true ) ;
} else {
return SMB_VFS_NEXT_STAT ( handle , smb_fname ) ;
}
if ( rc = = 0 ) {
update_btime ( handle , smb_fname ) ;
smb_fname - > st . st_ex_mode & = ~ S_IFMT ;
smb_fname - > st . st_ex_mode | = S_IFREG ;
smb_fname - > st . st_ex_blocks =
smb_fname - > st . st_ex_size / STAT_ST_BLOCKSIZE + 1 ;
}
return rc ;
}
static int fruit_lstat ( vfs_handle_struct * handle ,
struct smb_filename * smb_fname )
{
int rc = - 1 ;
DEBUG ( 10 , ( " fruit_lstat called for %s \n " ,
smb_fname_str_dbg ( smb_fname ) ) ) ;
2019-09-26 20:31:51 +03:00
if ( ! is_named_stream ( smb_fname ) ) {
2014-06-23 18:59:45 +04:00
rc = SMB_VFS_NEXT_LSTAT ( handle , smb_fname ) ;
if ( rc = = 0 ) {
update_btime ( handle , smb_fname ) ;
}
return rc ;
}
2019-07-01 16:53:36 +03:00
if ( is_afpinfo_stream ( smb_fname - > stream_name ) ) {
2014-06-23 18:59:45 +04:00
rc = fruit_stat_meta ( handle , smb_fname , false ) ;
2019-07-01 16:53:36 +03:00
} else if ( is_afpresource_stream ( smb_fname - > stream_name ) ) {
2014-06-23 18:59:45 +04:00
rc = fruit_stat_rsrc ( handle , smb_fname , false ) ;
} else {
return SMB_VFS_NEXT_LSTAT ( handle , smb_fname ) ;
}
if ( rc = = 0 ) {
update_btime ( handle , smb_fname ) ;
smb_fname - > st . st_ex_mode & = ~ S_IFMT ;
smb_fname - > st . st_ex_mode | = S_IFREG ;
smb_fname - > st . st_ex_blocks =
smb_fname - > st . st_ex_size / STAT_ST_BLOCKSIZE + 1 ;
}
return rc ;
}
2016-12-08 22:39:38 +03:00
static int fruit_fstat_meta_stream ( vfs_handle_struct * handle ,
files_struct * fsp ,
SMB_STRUCT_STAT * sbuf )
2014-06-23 18:59:45 +04:00
{
2020-12-10 15:11:06 +03:00
struct fio * fio = fruit_get_complete_fio ( handle , fsp ) ;
2018-11-30 12:27:19 +03:00
struct smb_filename smb_fname ;
2018-08-22 17:49:23 +03:00
ino_t ino ;
int ret ;
if ( fio = = NULL ) {
return - 1 ;
}
if ( fio - > fake_fd ) {
ret = fruit_stat_base ( handle , fsp - > base_fsp - > fsp_name , false ) ;
if ( ret ! = 0 ) {
return - 1 ;
}
* sbuf = fsp - > base_fsp - > fsp_name - > st ;
sbuf - > st_ex_size = AFP_INFO_SIZE ;
2019-06-18 17:58:29 +03:00
sbuf - > st_ex_ino = hash_inode ( sbuf , fsp - > fsp_name - > stream_name ) ;
2018-08-22 17:49:23 +03:00
return 0 ;
}
2018-11-30 12:27:19 +03:00
smb_fname = ( struct smb_filename ) {
. base_name = fsp - > fsp_name - > base_name ,
2020-05-02 11:28:44 +03:00
. twrp = fsp - > fsp_name - > twrp ,
2018-11-30 12:27:19 +03:00
} ;
ret = fruit_stat_base ( handle , & smb_fname , false ) ;
2018-08-22 17:49:23 +03:00
if ( ret ! = 0 ) {
return - 1 ;
}
2018-11-30 12:27:19 +03:00
* sbuf = smb_fname . st ;
2018-08-22 17:49:23 +03:00
2019-06-18 17:58:29 +03:00
ino = hash_inode ( sbuf , fsp - > fsp_name - > stream_name ) ;
2018-08-22 17:49:23 +03:00
ret = SMB_VFS_NEXT_FSTAT ( handle , fsp , sbuf ) ;
if ( ret ! = 0 ) {
return - 1 ;
}
sbuf - > st_ex_ino = ino ;
return 0 ;
2016-12-08 22:39:38 +03:00
}
2014-06-23 18:59:45 +04:00
2016-12-08 22:39:38 +03:00
static int fruit_fstat_meta_netatalk ( vfs_handle_struct * handle ,
files_struct * fsp ,
SMB_STRUCT_STAT * sbuf )
{
int ret ;
ret = fruit_stat_base ( handle , fsp - > base_fsp - > fsp_name , false ) ;
if ( ret ! = 0 ) {
2014-06-23 18:59:45 +04:00
return - 1 ;
}
2016-12-08 22:39:38 +03:00
2014-06-23 18:59:45 +04:00
* sbuf = fsp - > base_fsp - > fsp_name - > st ;
sbuf - > st_ex_size = AFP_INFO_SIZE ;
2019-06-18 17:58:29 +03:00
sbuf - > st_ex_ino = hash_inode ( sbuf , fsp - > fsp_name - > stream_name ) ;
2014-06-23 18:59:45 +04:00
return 0 ;
}
2016-12-08 22:39:38 +03:00
static int fruit_fstat_meta ( vfs_handle_struct * handle ,
files_struct * fsp ,
SMB_STRUCT_STAT * sbuf ,
struct fio * fio )
2014-06-23 18:59:45 +04:00
{
2016-12-08 22:39:38 +03:00
int ret ;
2014-06-23 18:59:45 +04:00
2016-12-08 22:39:38 +03:00
DBG_DEBUG ( " Path [%s] \n " , fsp_str_dbg ( fsp ) ) ;
2014-06-23 18:59:45 +04:00
2016-12-08 22:39:38 +03:00
switch ( fio - > config - > meta ) {
case FRUIT_META_STREAM :
ret = fruit_fstat_meta_stream ( handle , fsp , sbuf ) ;
break ;
2014-06-23 18:59:45 +04:00
2016-12-08 22:39:38 +03:00
case FRUIT_META_NETATALK :
ret = fruit_fstat_meta_netatalk ( handle , fsp , sbuf ) ;
break ;
default :
DBG_ERR ( " Unexpected meta config [%d] \n " , fio - > config - > meta ) ;
return - 1 ;
2014-06-23 18:59:45 +04:00
}
2016-12-08 22:39:38 +03:00
DBG_DEBUG ( " Path [%s] ret [%d] \n " , fsp_str_dbg ( fsp ) , ret ) ;
return ret ;
}
static int fruit_fstat_rsrc_xattr ( vfs_handle_struct * handle ,
files_struct * fsp ,
SMB_STRUCT_STAT * sbuf )
{
return SMB_VFS_NEXT_FSTAT ( handle , fsp , sbuf ) ;
}
static int fruit_fstat_rsrc_stream ( vfs_handle_struct * handle ,
files_struct * fsp ,
SMB_STRUCT_STAT * sbuf )
{
return SMB_VFS_NEXT_FSTAT ( handle , fsp , sbuf ) ;
}
static int fruit_fstat_rsrc_adouble ( vfs_handle_struct * handle ,
files_struct * fsp ,
SMB_STRUCT_STAT * sbuf )
{
2020-12-08 18:29:10 +03:00
struct fio * fio = fruit_get_complete_fio ( handle , fsp ) ;
2016-12-08 22:39:38 +03:00
struct adouble * ad = NULL ;
int ret ;
2022-01-10 15:26:25 +03:00
if ( fio = = NULL | | fio - > ad_fsp = = NULL ) {
DBG_ERR ( " fio/ad_fsp=NULL for [%s] \n " , fsp_str_dbg ( fsp ) ) ;
2020-12-08 18:29:10 +03:00
errno = EBADF ;
return - 1 ;
}
2014-06-23 18:59:45 +04:00
/* Populate the stat struct with info from the base file. */
2016-12-08 22:39:38 +03:00
ret = fruit_stat_base ( handle , fsp - > base_fsp - > fsp_name , false ) ;
if ( ret = = - 1 ) {
return - 1 ;
}
2020-12-08 18:29:10 +03:00
ad = ad_fget ( talloc_tos ( ) , handle , fio - > ad_fsp , ADOUBLE_RSRC ) ;
2016-12-08 22:39:38 +03:00
if ( ad = = NULL ) {
2020-12-08 18:29:10 +03:00
DBG_ERR ( " ad_fget [%s] failed [%s] \n " ,
fsp_str_dbg ( fio - > ad_fsp ) , strerror ( errno ) ) ;
2014-06-23 18:59:45 +04:00
return - 1 ;
}
2016-12-08 22:39:38 +03:00
2014-06-23 18:59:45 +04:00
* sbuf = fsp - > base_fsp - > fsp_name - > st ;
sbuf - > st_ex_size = ad_getentrylen ( ad , ADEID_RFORK ) ;
2019-06-18 17:58:29 +03:00
sbuf - > st_ex_ino = hash_inode ( sbuf , fsp - > fsp_name - > stream_name ) ;
2014-06-23 18:59:45 +04:00
2016-12-08 22:39:38 +03:00
TALLOC_FREE ( ad ) ;
2014-06-23 18:59:45 +04:00
return 0 ;
}
2016-12-08 22:39:38 +03:00
static int fruit_fstat_rsrc ( vfs_handle_struct * handle , files_struct * fsp ,
SMB_STRUCT_STAT * sbuf , struct fio * fio )
{
int ret ;
switch ( fio - > config - > rsrc ) {
case FRUIT_RSRC_STREAM :
ret = fruit_fstat_rsrc_stream ( handle , fsp , sbuf ) ;
break ;
case FRUIT_RSRC_ADFILE :
ret = fruit_fstat_rsrc_adouble ( handle , fsp , sbuf ) ;
break ;
case FRUIT_RSRC_XATTR :
ret = fruit_fstat_rsrc_xattr ( handle , fsp , sbuf ) ;
break ;
default :
DBG_ERR ( " Unexpected rsrc config [%d] \n " , fio - > config - > rsrc ) ;
return - 1 ;
}
return ret ;
}
2014-06-23 18:59:45 +04:00
static int fruit_fstat ( vfs_handle_struct * handle , files_struct * fsp ,
SMB_STRUCT_STAT * sbuf )
{
2020-12-10 15:11:06 +03:00
struct fio * fio = fruit_get_complete_fio ( handle , fsp ) ;
2014-06-23 18:59:45 +04:00
int rc ;
2016-12-08 22:39:38 +03:00
if ( fio = = NULL ) {
return SMB_VFS_NEXT_FSTAT ( handle , fsp , sbuf ) ;
2014-06-23 18:59:45 +04:00
}
2016-12-08 22:39:38 +03:00
DBG_DEBUG ( " Path [%s] \n " , fsp_str_dbg ( fsp ) ) ;
if ( fio - > type = = ADOUBLE_META ) {
rc = fruit_fstat_meta ( handle , fsp , sbuf , fio ) ;
} else {
rc = fruit_fstat_rsrc ( handle , fsp , sbuf , fio ) ;
2014-06-23 18:59:45 +04:00
}
if ( rc = = 0 ) {
sbuf - > st_ex_mode & = ~ S_IFMT ;
sbuf - > st_ex_mode | = S_IFREG ;
sbuf - > st_ex_blocks = sbuf - > st_ex_size / STAT_ST_BLOCKSIZE + 1 ;
}
2017-04-29 13:01:41 +03:00
DBG_DEBUG ( " Path [%s] rc [%d] size [% " PRIdMAX " ] \n " ,
fsp_str_dbg ( fsp ) , rc , ( intmax_t ) sbuf - > st_ex_size ) ;
2014-06-23 18:59:45 +04:00
return rc ;
}
2017-12-07 16:56:36 +03:00
static NTSTATUS delete_invalid_meta_stream (
2016-12-02 17:49:03 +03:00
vfs_handle_struct * handle ,
const struct smb_filename * smb_fname ,
TALLOC_CTX * mem_ctx ,
unsigned int * pnum_streams ,
2018-10-21 00:46:43 +03:00
struct stream_struct * * pstreams ,
off_t size )
2016-12-02 17:49:03 +03:00
{
2016-12-11 21:10:05 +03:00
struct smb_filename * sname = NULL ;
2021-01-21 16:37:53 +03:00
NTSTATUS status ;
2016-12-11 21:10:05 +03:00
int ret ;
bool ok ;
ok = del_fruit_stream ( mem_ctx , pnum_streams , pstreams , AFPINFO_STREAM ) ;
if ( ! ok ) {
return NT_STATUS_INTERNAL_ERROR ;
}
2018-10-21 00:46:43 +03:00
if ( size = = 0 ) {
return NT_STATUS_OK ;
}
2021-01-21 16:37:53 +03:00
status = synthetic_pathref ( talloc_tos ( ) ,
handle - > conn - > cwd_fsp ,
smb_fname - > base_name ,
AFPINFO_STREAM_NAME ,
NULL ,
smb_fname - > twrp ,
0 ,
& sname ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2016-12-11 21:10:05 +03:00
return NT_STATUS_NO_MEMORY ;
}
2019-09-14 00:21:28 +03:00
ret = SMB_VFS_NEXT_UNLINKAT ( handle ,
handle - > conn - > cwd_fsp ,
sname ,
0 ) ;
2016-12-11 21:10:05 +03:00
if ( ret ! = 0 ) {
DBG_ERR ( " Removing [%s] failed \n " , smb_fname_str_dbg ( sname ) ) ;
2021-01-21 16:36:59 +03:00
TALLOC_FREE ( sname ) ;
2016-12-11 21:10:05 +03:00
return map_nt_error_from_unix ( errno ) ;
}
2021-01-21 16:36:59 +03:00
TALLOC_FREE ( sname ) ;
2016-12-02 17:49:03 +03:00
return NT_STATUS_OK ;
}
2017-12-07 16:56:36 +03:00
static NTSTATUS fruit_streaminfo_meta_stream (
vfs_handle_struct * handle ,
struct files_struct * fsp ,
const struct smb_filename * smb_fname ,
TALLOC_CTX * mem_ctx ,
unsigned int * pnum_streams ,
struct stream_struct * * pstreams )
{
struct stream_struct * stream = * pstreams ;
2024-11-04 20:19:48 +03:00
unsigned int i , num_streams = * pnum_streams ;
2017-12-07 16:56:36 +03:00
for ( i = 0 ; i < num_streams ; i + + ) {
if ( strequal_m ( stream [ i ] . name , AFPINFO_STREAM ) ) {
break ;
}
}
if ( i = = num_streams ) {
return NT_STATUS_OK ;
}
if ( stream [ i ] . size ! = AFP_INFO_SIZE ) {
DBG_ERR ( " Removing invalid AFPINFO_STREAM size [%jd] from [%s] \n " ,
( intmax_t ) stream [ i ] . size , smb_fname_str_dbg ( smb_fname ) ) ;
2018-10-21 00:46:43 +03:00
return delete_invalid_meta_stream ( handle ,
smb_fname ,
mem_ctx ,
pnum_streams ,
pstreams ,
stream [ i ] . size ) ;
2017-12-07 16:56:36 +03:00
}
2018-08-22 17:49:23 +03:00
2018-10-21 00:50:32 +03:00
return NT_STATUS_OK ;
2017-12-07 16:56:36 +03:00
}
2016-12-02 17:49:03 +03:00
static NTSTATUS fruit_streaminfo_meta_netatalk (
vfs_handle_struct * handle ,
struct files_struct * fsp ,
const struct smb_filename * smb_fname ,
TALLOC_CTX * mem_ctx ,
unsigned int * pnum_streams ,
struct stream_struct * * pstreams )
{
2016-12-11 21:10:05 +03:00
struct stream_struct * stream = * pstreams ;
2024-11-04 20:19:48 +03:00
unsigned int i , num_streams = * pnum_streams ;
2016-12-02 17:49:03 +03:00
struct adouble * ad = NULL ;
bool is_fi_empty ;
bool ok ;
/* Remove the Netatalk xattr from the list */
ok = del_fruit_stream ( mem_ctx , pnum_streams , pstreams ,
" : " NETATALK_META_XATTR " :$DATA " ) ;
if ( ! ok ) {
return NT_STATUS_NO_MEMORY ;
}
2016-12-11 21:10:05 +03:00
/*
* Check if there ' s a AFPINFO_STREAM from the VFS streams
* backend and if yes , remove it from the list
*/
for ( i = 0 ; i < num_streams ; i + + ) {
if ( strequal_m ( stream [ i ] . name , AFPINFO_STREAM ) ) {
break ;
}
}
if ( i < num_streams ) {
DBG_WARNING ( " Unexpected AFPINFO_STREAM on [%s] \n " ,
smb_fname_str_dbg ( smb_fname ) ) ;
ok = del_fruit_stream ( mem_ctx , pnum_streams , pstreams ,
AFPINFO_STREAM ) ;
if ( ! ok ) {
return NT_STATUS_INTERNAL_ERROR ;
}
}
2021-07-02 23:55:04 +03:00
ad = ad_get_meta_fsp ( talloc_tos ( ) , handle , smb_fname ) ;
2016-12-02 17:49:03 +03:00
if ( ad = = NULL ) {
return NT_STATUS_OK ;
}
is_fi_empty = ad_empty_finderinfo ( ad ) ;
TALLOC_FREE ( ad ) ;
if ( is_fi_empty ) {
return NT_STATUS_OK ;
}
ok = add_fruit_stream ( mem_ctx , pnum_streams , pstreams ,
AFPINFO_STREAM_NAME , AFP_INFO_SIZE ,
smb_roundup ( handle - > conn , AFP_INFO_SIZE ) ) ;
if ( ! ok ) {
return NT_STATUS_NO_MEMORY ;
}
return NT_STATUS_OK ;
}
static NTSTATUS fruit_streaminfo_meta ( vfs_handle_struct * handle ,
struct files_struct * fsp ,
const struct smb_filename * smb_fname ,
TALLOC_CTX * mem_ctx ,
unsigned int * pnum_streams ,
struct stream_struct * * pstreams )
{
struct fruit_config_data * config = NULL ;
NTSTATUS status ;
SMB_VFS_HANDLE_GET_DATA ( handle , config , struct fruit_config_data ,
return NT_STATUS_INTERNAL_ERROR ) ;
switch ( config - > meta ) {
case FRUIT_META_NETATALK :
status = fruit_streaminfo_meta_netatalk ( handle , fsp , smb_fname ,
mem_ctx , pnum_streams ,
pstreams ) ;
break ;
case FRUIT_META_STREAM :
status = fruit_streaminfo_meta_stream ( handle , fsp , smb_fname ,
mem_ctx , pnum_streams ,
pstreams ) ;
break ;
default :
return NT_STATUS_INTERNAL_ERROR ;
}
return status ;
}
static NTSTATUS fruit_streaminfo_rsrc_stream (
vfs_handle_struct * handle ,
struct files_struct * fsp ,
const struct smb_filename * smb_fname ,
TALLOC_CTX * mem_ctx ,
unsigned int * pnum_streams ,
struct stream_struct * * pstreams )
{
2024-11-04 20:17:16 +03:00
filter_empty_rsrc_stream ( pnum_streams , pstreams ) ;
2016-12-02 17:49:03 +03:00
return NT_STATUS_OK ;
}
static NTSTATUS fruit_streaminfo_rsrc_xattr (
vfs_handle_struct * handle ,
struct files_struct * fsp ,
const struct smb_filename * smb_fname ,
TALLOC_CTX * mem_ctx ,
unsigned int * pnum_streams ,
struct stream_struct * * pstreams )
{
2024-11-04 20:17:16 +03:00
filter_empty_rsrc_stream ( pnum_streams , pstreams ) ;
2016-12-02 17:49:03 +03:00
return NT_STATUS_OK ;
}
static NTSTATUS fruit_streaminfo_rsrc_adouble (
vfs_handle_struct * handle ,
struct files_struct * fsp ,
const struct smb_filename * smb_fname ,
TALLOC_CTX * mem_ctx ,
unsigned int * pnum_streams ,
struct stream_struct * * pstreams )
{
struct stream_struct * stream = * pstreams ;
2024-11-04 20:19:48 +03:00
unsigned int i , num_streams = * pnum_streams ;
2016-12-02 17:49:03 +03:00
struct adouble * ad = NULL ;
bool ok ;
size_t rlen ;
/*
* Check if there ' s a AFPRESOURCE_STREAM from the VFS streams backend
* and if yes , remove it from the list
*/
for ( i = 0 ; i < num_streams ; i + + ) {
if ( strequal_m ( stream [ i ] . name , AFPRESOURCE_STREAM ) ) {
break ;
}
}
if ( i < num_streams ) {
DBG_WARNING ( " Unexpected AFPRESOURCE_STREAM on [%s] \n " ,
smb_fname_str_dbg ( smb_fname ) ) ;
ok = del_fruit_stream ( mem_ctx , pnum_streams , pstreams ,
AFPRESOURCE_STREAM ) ;
if ( ! ok ) {
return NT_STATUS_INTERNAL_ERROR ;
}
}
2017-05-25 21:38:26 +03:00
ad = ad_get ( talloc_tos ( ) , handle , smb_fname , ADOUBLE_RSRC ) ;
2016-12-02 17:49:03 +03:00
if ( ad = = NULL ) {
return NT_STATUS_OK ;
}
rlen = ad_getentrylen ( ad , ADEID_RFORK ) ;
TALLOC_FREE ( ad ) ;
if ( rlen = = 0 ) {
return NT_STATUS_OK ;
}
ok = add_fruit_stream ( mem_ctx , pnum_streams , pstreams ,
AFPRESOURCE_STREAM_NAME , rlen ,
smb_roundup ( handle - > conn , rlen ) ) ;
if ( ! ok ) {
return NT_STATUS_NO_MEMORY ;
}
return NT_STATUS_OK ;
}
static NTSTATUS fruit_streaminfo_rsrc ( vfs_handle_struct * handle ,
struct files_struct * fsp ,
const struct smb_filename * smb_fname ,
TALLOC_CTX * mem_ctx ,
unsigned int * pnum_streams ,
struct stream_struct * * pstreams )
{
struct fruit_config_data * config = NULL ;
NTSTATUS status ;
2023-05-22 20:37:17 +03:00
if ( S_ISDIR ( smb_fname - > st . st_ex_mode ) ) {
return NT_STATUS_OK ;
}
2016-12-02 17:49:03 +03:00
SMB_VFS_HANDLE_GET_DATA ( handle , config , struct fruit_config_data ,
return NT_STATUS_INTERNAL_ERROR ) ;
switch ( config - > rsrc ) {
case FRUIT_RSRC_STREAM :
status = fruit_streaminfo_rsrc_stream ( handle , fsp , smb_fname ,
mem_ctx , pnum_streams ,
pstreams ) ;
break ;
case FRUIT_RSRC_XATTR :
status = fruit_streaminfo_rsrc_xattr ( handle , fsp , smb_fname ,
mem_ctx , pnum_streams ,
pstreams ) ;
break ;
case FRUIT_RSRC_ADFILE :
status = fruit_streaminfo_rsrc_adouble ( handle , fsp , smb_fname ,
mem_ctx , pnum_streams ,
pstreams ) ;
break ;
default :
return NT_STATUS_INTERNAL_ERROR ;
}
return status ;
}
2018-10-20 15:53:50 +03:00
static void fruit_filter_empty_streams ( unsigned int * pnum_streams ,
struct stream_struct * * pstreams )
{
unsigned num_streams = * pnum_streams ;
struct stream_struct * streams = * pstreams ;
unsigned i = 0 ;
if ( ! global_fruit_config . nego_aapl ) {
return ;
}
while ( i < num_streams ) {
struct smb_filename smb_fname = ( struct smb_filename ) {
. stream_name = streams [ i ] . name ,
} ;
if ( is_ntfs_default_stream_smb_fname ( & smb_fname )
| | streams [ i ] . size > 0 )
{
i + + ;
continue ;
}
streams [ i ] = streams [ num_streams - 1 ] ;
num_streams - - ;
}
* pnum_streams = num_streams ;
}
2021-04-27 18:09:35 +03:00
static NTSTATUS fruit_fstreaminfo ( vfs_handle_struct * handle ,
struct files_struct * fsp ,
TALLOC_CTX * mem_ctx ,
unsigned int * pnum_streams ,
struct stream_struct * * pstreams )
{
struct fruit_config_data * config = NULL ;
const struct smb_filename * smb_fname = NULL ;
NTSTATUS status ;
smb_fname = fsp - > fsp_name ;
SMB_VFS_HANDLE_GET_DATA ( handle , config , struct fruit_config_data ,
return NT_STATUS_UNSUCCESSFUL ) ;
DBG_DEBUG ( " Path [%s] \n " , smb_fname_str_dbg ( smb_fname ) ) ;
status = SMB_VFS_NEXT_FSTREAMINFO ( handle , fsp , mem_ctx ,
pnum_streams , pstreams ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
}
fruit_filter_empty_streams ( pnum_streams , pstreams ) ;
status = fruit_streaminfo_meta ( handle , fsp , smb_fname ,
mem_ctx , pnum_streams , pstreams ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
}
status = fruit_streaminfo_rsrc ( handle , fsp , smb_fname ,
mem_ctx , pnum_streams , pstreams ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
}
return NT_STATUS_OK ;
}
2021-04-13 17:23:41 +03:00
static int fruit_fntimes ( vfs_handle_struct * handle ,
files_struct * fsp ,
struct smb_file_time * ft )
{
int rc = 0 ;
struct adouble * ad = NULL ;
struct fruit_config_data * config = NULL ;
SMB_VFS_HANDLE_GET_DATA ( handle , config , struct fruit_config_data ,
return - 1 ) ;
if ( ( config - > meta ! = FRUIT_META_NETATALK ) | |
is_omit_timespec ( & ft - > create_time ) )
{
return SMB_VFS_NEXT_FNTIMES ( handle , fsp , ft ) ;
}
2023-08-07 07:43:37 +03:00
DBG_DEBUG ( " set btime for %s to %s " , fsp_str_dbg ( fsp ) ,
2021-04-13 17:23:41 +03:00
time_to_asc ( convert_timespec_to_time_t ( ft - > create_time ) ) ) ;
ad = ad_fget ( talloc_tos ( ) , handle , fsp , ADOUBLE_META ) ;
if ( ad = = NULL ) {
goto exit ;
}
ad_setdate ( ad , AD_DATE_CREATE | AD_DATE_UNIX ,
convert_time_t_to_uint32_t ( ft - > create_time . tv_sec ) ) ;
rc = ad_fset ( handle , ad , fsp ) ;
exit :
TALLOC_FREE ( ad ) ;
if ( rc ! = 0 ) {
DBG_WARNING ( " %s \n " , fsp_str_dbg ( fsp ) ) ;
return - 1 ;
}
return SMB_VFS_NEXT_FNTIMES ( handle , fsp , ft ) ;
}
2014-06-23 18:59:45 +04:00
static int fruit_fallocate ( struct vfs_handle_struct * handle ,
struct files_struct * fsp ,
2015-02-09 20:21:59 +03:00
uint32_t mode ,
2014-06-23 18:59:45 +04:00
off_t offset ,
off_t len )
{
2020-12-10 15:11:06 +03:00
struct fio * fio = fruit_get_complete_fio ( handle , fsp ) ;
2014-06-23 18:59:45 +04:00
2016-12-08 22:41:55 +03:00
if ( fio = = NULL ) {
2014-06-23 18:59:45 +04:00
return SMB_VFS_NEXT_FALLOCATE ( handle , fsp , mode , offset , len ) ;
}
/* Let the pwrite code path handle it. */
2014-12-06 02:37:11 +03:00
errno = ENOSYS ;
return - 1 ;
2014-06-23 18:59:45 +04:00
}
2016-12-02 18:20:46 +03:00
static int fruit_ftruncate_rsrc_xattr ( struct vfs_handle_struct * handle ,
struct files_struct * fsp ,
off_t offset )
{
# ifdef HAVE_ATTROPEN
return SMB_VFS_NEXT_FTRUNCATE ( handle , fsp , offset ) ;
# endif
return 0 ;
}
static int fruit_ftruncate_rsrc_adouble ( struct vfs_handle_struct * handle ,
struct files_struct * fsp ,
off_t offset )
{
2020-12-08 18:29:10 +03:00
struct fio * fio = fruit_get_complete_fio ( handle , fsp ) ;
2016-12-02 18:20:46 +03:00
int rc ;
2016-12-08 21:12:32 +03:00
struct adouble * ad = NULL ;
off_t ad_off ;
2016-12-02 18:20:46 +03:00
2022-01-10 15:26:25 +03:00
if ( fio = = NULL | | fio - > ad_fsp = = NULL ) {
DBG_ERR ( " fio/ad_fsp=NULL for [%s] \n " , fsp_str_dbg ( fsp ) ) ;
2020-12-08 18:29:10 +03:00
errno = EBADF ;
return - 1 ;
}
ad = ad_fget ( talloc_tos ( ) , handle , fio - > ad_fsp , ADOUBLE_RSRC ) ;
2016-12-08 21:12:32 +03:00
if ( ad = = NULL ) {
2020-12-08 18:29:10 +03:00
DBG_ERR ( " ad_fget [%s] failed [%s] \n " ,
fsp_str_dbg ( fio - > ad_fsp ) , strerror ( errno ) ) ;
2016-12-02 18:20:46 +03:00
return - 1 ;
}
2016-12-08 21:12:32 +03:00
ad_off = ad_getentryoff ( ad , ADEID_RFORK ) ;
2020-12-08 18:29:10 +03:00
rc = SMB_VFS_NEXT_FTRUNCATE ( handle , fio - > ad_fsp , offset + ad_off ) ;
2016-12-02 18:20:46 +03:00
if ( rc ! = 0 ) {
2016-12-08 22:42:54 +03:00
TALLOC_FREE ( ad ) ;
2016-12-02 18:20:46 +03:00
return - 1 ;
}
ad_setentrylen ( ad , ADEID_RFORK , offset ) ;
2020-12-08 18:29:10 +03:00
rc = ad_fset ( handle , ad , fio - > ad_fsp ) ;
2016-12-02 18:20:46 +03:00
if ( rc ! = 0 ) {
2016-12-08 22:42:54 +03:00
DBG_ERR ( " ad_fset [%s] failed [%s] \n " ,
2020-12-08 18:29:10 +03:00
fsp_str_dbg ( fio - > ad_fsp ) , strerror ( errno ) ) ;
2016-12-08 22:42:54 +03:00
TALLOC_FREE ( ad ) ;
2016-12-02 18:20:46 +03:00
return - 1 ;
}
2016-12-08 22:42:54 +03:00
TALLOC_FREE ( ad ) ;
2016-12-02 18:20:46 +03:00
return 0 ;
}
static int fruit_ftruncate_rsrc_stream ( struct vfs_handle_struct * handle ,
struct files_struct * fsp ,
off_t offset )
{
return SMB_VFS_NEXT_FTRUNCATE ( handle , fsp , offset ) ;
}
2015-08-12 08:34:53 +03:00
static int fruit_ftruncate_rsrc ( struct vfs_handle_struct * handle ,
struct files_struct * fsp ,
2016-12-02 18:20:46 +03:00
off_t offset )
2015-08-12 08:34:53 +03:00
{
2020-12-10 15:11:06 +03:00
struct fio * fio = fruit_get_complete_fio ( handle , fsp ) ;
2016-12-02 18:20:46 +03:00
int ret ;
2015-08-12 08:34:53 +03:00
2018-05-25 13:43:42 +03:00
if ( fio = = NULL ) {
2023-08-07 07:37:51 +03:00
DBG_ERR ( " Failed to fetch fsp extension \n " ) ;
2018-05-25 13:43:42 +03:00
return - 1 ;
}
2016-12-08 22:42:54 +03:00
switch ( fio - > config - > rsrc ) {
2016-12-02 18:20:46 +03:00
case FRUIT_RSRC_XATTR :
ret = fruit_ftruncate_rsrc_xattr ( handle , fsp , offset ) ;
break ;
2015-08-12 08:34:53 +03:00
2016-12-02 18:20:46 +03:00
case FRUIT_RSRC_ADFILE :
ret = fruit_ftruncate_rsrc_adouble ( handle , fsp , offset ) ;
break ;
case FRUIT_RSRC_STREAM :
ret = fruit_ftruncate_rsrc_stream ( handle , fsp , offset ) ;
break ;
default :
2016-12-08 22:42:54 +03:00
DBG_ERR ( " Unexpected rsrc config [%d] \n " , fio - > config - > rsrc ) ;
2015-08-12 08:34:53 +03:00
return - 1 ;
}
2016-12-02 18:20:46 +03:00
return ret ;
2015-08-12 08:34:53 +03:00
}
2016-12-08 22:42:54 +03:00
static int fruit_ftruncate_meta ( struct vfs_handle_struct * handle ,
struct files_struct * fsp ,
off_t offset )
{
if ( offset > 60 ) {
2023-08-07 07:37:51 +03:00
DBG_WARNING ( " ftruncate %s to %jd \n " ,
2016-12-08 22:42:54 +03:00
fsp_str_dbg ( fsp ) , ( intmax_t ) offset ) ;
/* OS X returns NT_STATUS_ALLOTTED_SPACE_EXCEEDED */
errno = EOVERFLOW ;
return - 1 ;
}
/* OS X returns success but does nothing */
DBG_INFO ( " ignoring ftruncate %s to %jd \n " ,
fsp_str_dbg ( fsp ) , ( intmax_t ) offset ) ;
return 0 ;
}
2014-06-23 18:59:45 +04:00
static int fruit_ftruncate ( struct vfs_handle_struct * handle ,
struct files_struct * fsp ,
off_t offset )
{
2020-12-10 15:11:06 +03:00
struct fio * fio = fruit_get_complete_fio ( handle , fsp ) ;
2016-12-08 22:42:54 +03:00
int ret ;
2016-11-15 22:32:05 +03:00
2017-04-29 13:01:41 +03:00
DBG_DEBUG ( " Path [%s] offset [% " PRIdMAX " ] \n " , fsp_str_dbg ( fsp ) ,
( intmax_t ) offset ) ;
2016-12-08 22:42:54 +03:00
if ( fio = = NULL ) {
return SMB_VFS_NEXT_FTRUNCATE ( handle , fsp , offset ) ;
2016-11-15 22:32:05 +03:00
}
2016-12-08 22:42:54 +03:00
if ( fio - > type = = ADOUBLE_META ) {
ret = fruit_ftruncate_meta ( handle , fsp , offset ) ;
} else {
ret = fruit_ftruncate_rsrc ( handle , fsp , offset ) ;
2014-06-23 18:59:45 +04:00
}
2016-12-08 22:42:54 +03:00
DBG_DEBUG ( " Path [%s] result [%d] \n " , fsp_str_dbg ( fsp ) , ret ) ;
return ret ;
2014-06-23 18:59:45 +04:00
}
static NTSTATUS fruit_create_file ( vfs_handle_struct * handle ,
struct smb_request * req ,
2021-11-23 14:29:17 +03:00
struct files_struct * dirfsp ,
2014-06-23 18:59:45 +04:00
struct smb_filename * smb_fname ,
uint32_t access_mask ,
uint32_t share_access ,
uint32_t create_disposition ,
uint32_t create_options ,
uint32_t file_attributes ,
uint32_t oplock_request ,
2019-08-07 23:00:11 +03:00
const struct smb2_lease * lease ,
2014-06-23 18:59:45 +04:00
uint64_t allocation_size ,
uint32_t private_flags ,
struct security_descriptor * sd ,
struct ea_list * ea_list ,
files_struct * * result ,
2014-11-26 16:12:51 +03:00
int * pinfo ,
const struct smb2_create_blobs * in_context_blobs ,
struct smb2_create_blobs * out_context_blobs )
2014-06-23 18:59:45 +04:00
{
NTSTATUS status ;
struct fruit_config_data * config = NULL ;
2015-08-08 21:21:39 +03:00
files_struct * fsp = NULL ;
2019-05-21 17:00:53 +03:00
bool internal_open = ( oplock_request & INTERNAL_OPEN_ONLY ) ;
int ret ;
2014-06-23 18:59:45 +04:00
2014-11-26 20:11:17 +03:00
status = check_aapl ( handle , req , in_context_blobs , out_context_blobs ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2015-09-08 09:18:02 +03:00
goto fail ;
2014-11-26 20:11:17 +03:00
}
SMB_VFS_HANDLE_GET_DATA ( handle , config , struct fruit_config_data ,
return NT_STATUS_UNSUCCESSFUL ) ;
2023-05-22 13:32:00 +03:00
if ( is_apple_stream ( smb_fname - > stream_name ) & &
! internal_open & &
config - > convert_adouble )
{
2019-05-27 17:57:51 +03:00
uint32_t conv_flags = 0 ;
if ( config - > wipe_intentionally_left_blank_rfork ) {
conv_flags | = AD_CONV_WIPE_BLANK ;
}
if ( config - > delete_empty_adfiles ) {
conv_flags | = AD_CONV_DELETE ;
}
2019-05-27 18:55:07 +03:00
ret = ad_convert ( handle ,
smb_fname ,
2019-07-09 20:26:01 +03:00
macos_string_replace_map ,
2019-05-27 18:55:07 +03:00
conv_flags ) ;
2019-05-21 17:00:53 +03:00
if ( ret ! = 0 ) {
2023-05-22 13:25:04 +03:00
DBG_ERR ( " ad_convert( \" %s \" ) failed \n " ,
smb_fname_str_dbg ( smb_fname ) ) ;
2019-05-21 17:00:53 +03:00
}
}
2014-06-23 18:59:45 +04:00
status = SMB_VFS_NEXT_CREATE_FILE (
2021-11-23 14:29:17 +03:00
handle , req , dirfsp , smb_fname ,
2014-06-23 18:59:45 +04:00
access_mask , share_access ,
create_disposition , create_options ,
file_attributes , oplock_request ,
lease ,
allocation_size , private_flags ,
sd , ea_list , result ,
2014-11-26 20:11:17 +03:00
pinfo , in_context_blobs , out_context_blobs ) ;
2014-06-23 18:59:45 +04:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
}
2015-11-25 11:12:55 +03:00
2015-08-08 21:21:39 +03:00
fsp = * result ;
2014-06-23 18:59:45 +04:00
2015-08-08 21:21:39 +03:00
/*
* If this is a plain open for existing files , opening an 0
* byte size resource fork MUST fail with
* NT_STATUS_OBJECT_NAME_NOT_FOUND .
*
* Cf the vfs_fruit torture tests in test_rfork_create ( ) .
*/
2018-10-20 16:28:06 +03:00
if ( global_fruit_config . nego_aapl & &
2018-10-20 15:53:50 +03:00
create_disposition = = FILE_OPEN & &
smb_fname - > st . st_ex_size = = 0 & &
2019-09-26 20:31:51 +03:00
is_named_stream ( smb_fname ) )
2015-08-08 21:21:39 +03:00
{
2018-10-20 15:53:50 +03:00
status = NT_STATUS_OBJECT_NAME_NOT_FOUND ;
goto fail ;
2015-04-22 23:29:16 +03:00
}
2015-08-08 21:21:39 +03:00
2020-04-02 19:21:11 +03:00
if ( is_named_stream ( smb_fname ) | | fsp - > fsp_flags . is_directory ) {
2014-06-23 18:59:45 +04:00
return status ;
}
2019-05-23 17:22:39 +03:00
if ( ( config - > locking = = FRUIT_LOCKING_NETATALK ) & &
2020-08-01 17:19:20 +03:00
( fsp - > op ! = NULL ) & &
! fsp - > fsp_flags . is_pathref )
2019-05-23 17:22:39 +03:00
{
2014-06-23 18:59:45 +04:00
status = fruit_check_access (
handle , * result ,
access_mask ,
s3: VFS: vfs_fruit. Fix the NetAtalk deny mode compatibility code.
This exhibited itself as a problem with OFD locks reported
as:
BUG: https://bugzilla.samba.org/show_bug.cgi?id=13770
However, due to underlying bugs in the vfs_fruit
code the file locks were not being properly applied.
There are two problems in fruit_check_access().
Problem #1:
Inside fruit_check_access() we have:
flags = fcntl(fsp->fh->fd, F_GETFL);
..
if (flags & (O_RDONLY|O_RDWR)) {
We shouldn't be calling fcntl(fsp->fh->fd, ..) directly.
fsp->fh->fd may be a made up number from an underlying
VFS module that has no meaning to a system call.
Secondly, in all POSIX systems - O_RDONLY is defined as
*zero*. O_RDWR = 2.
Which means flags & (O_RDONLY|O_RDWR) becomes (flags & 2),
not what we actually thought.
Problem #2:
deny_mode is *not* a bitmask, it's a set of discrete values.
Inside fruit_check_access() we have:
if (deny_mode & DENY_READ) and also (deny_mode & DENY_WRITE)
However, deny modes are defined as:
/* deny modes */
define DENY_DOS 0
define DENY_ALL 1
define DENY_WRITE 2
define DENY_READ 3
define DENY_NONE 4
define DENY_FCB 7
so if deny_mode = DENY_WRITE, or if deny_mode = DENY_READ
then it's going to trigger both the if (deny_mode & DENY_READ)
*and* the (deny_mode & DENY_WRITE) conditions.
These problems allowed the original test test_netatalk_lock code to
pass (which was added for BUG: https://bugzilla.samba.org/show_bug.cgi?id=13584
to demonstrate the lock order violation).
This patch refactors the fruit_check_access()
code to be much simpler (IMHO) to understand.
Firstly, pass in the SMB1/2 share mode, not old
DOS deny modes.
Secondly, read all the possible NetAtalk locks
into local variables:
netatalk_already_open_for_reading
netatalk_already_open_with_deny_read
netatalk_already_open_for_writing
netatalk_already_open_with_deny_write
Then do the share mode/access mode checks
with the requested values against any stored
netatalk modes/access modes.
Finally add in NetATalk compatible locks
that represent our share modes/access modes
into the file, with an early return if we don't
have FILE_READ_DATA (in which case we can't
write locks anyway).
The patch is easier to understand by looking
at the completed patched fruit_check_access()
function, rather than trying to look at the
diff.
Signed-off-by: Jeremy Allison <jra@samba.org>
Reviewed-by: Ralph Böhme <slow@samba.org>
2019-02-07 04:49:16 +03:00
share_access ) ;
2014-06-23 18:59:45 +04:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
goto fail ;
}
}
return status ;
fail :
2015-08-08 21:21:39 +03:00
DEBUG ( 10 , ( " fruit_create_file: %s \n " , nt_errstr ( status ) ) ) ;
2014-06-23 18:59:45 +04:00
2015-08-08 21:21:39 +03:00
if ( fsp ) {
2022-02-01 19:47:29 +03:00
close_file_free ( req , & fsp , ERROR_CLOSE ) ;
* result = NULL ;
2014-06-23 18:59:45 +04:00
}
return status ;
}
2021-05-13 14:08:20 +03:00
static NTSTATUS fruit_freaddir_attr ( struct vfs_handle_struct * handle ,
struct files_struct * fsp ,
TALLOC_CTX * mem_ctx ,
struct readdir_attr_data * * pattr_data )
{
struct fruit_config_data * config = NULL ;
struct readdir_attr_data * attr_data ;
uint32_t conv_flags = 0 ;
NTSTATUS status ;
int ret ;
SMB_VFS_HANDLE_GET_DATA ( handle , config ,
struct fruit_config_data ,
return NT_STATUS_UNSUCCESSFUL ) ;
if ( ! global_fruit_config . nego_aapl ) {
return SMB_VFS_NEXT_FREADDIR_ATTR ( handle ,
fsp ,
mem_ctx ,
pattr_data ) ;
}
DBG_DEBUG ( " Path [%s] \n " , fsp_str_dbg ( fsp ) ) ;
2023-05-22 13:32:00 +03:00
if ( config - > convert_adouble ) {
if ( config - > wipe_intentionally_left_blank_rfork ) {
conv_flags | = AD_CONV_WIPE_BLANK ;
}
if ( config - > delete_empty_adfiles ) {
conv_flags | = AD_CONV_DELETE ;
}
2021-05-13 14:08:20 +03:00
2023-05-22 13:32:00 +03:00
ret = ad_convert ( handle ,
fsp - > fsp_name ,
macos_string_replace_map ,
conv_flags ) ;
if ( ret ! = 0 ) {
DBG_ERR ( " ad_convert( \" %s \" ) failed \n " ,
fsp_str_dbg ( fsp ) ) ;
}
2021-05-13 14:08:20 +03:00
}
* pattr_data = talloc_zero ( mem_ctx , struct readdir_attr_data ) ;
if ( * pattr_data = = NULL ) {
return NT_STATUS_NO_MEMORY ;
}
attr_data = * pattr_data ;
attr_data - > type = RDATTR_AAPL ;
/*
* Mac metadata : compressed FinderInfo , resource fork length
* and creation date
*/
status = readdir_attr_macmeta ( handle , fsp - > fsp_name , attr_data ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
/*
* Error handling is tricky : if we return failure from
* this function , the corresponding directory entry
* will to be passed to the client , so we really just
* want to error out on fatal errors .
*/
if ( ! NT_STATUS_EQUAL ( status , NT_STATUS_ACCESS_DENIED ) ) {
goto fail ;
}
}
/*
* UNIX mode
*/
if ( config - > unix_info_enabled ) {
attr_data - > attr_data . aapl . unix_mode =
fsp - > fsp_name - > st . st_ex_mode ;
}
/*
* max_access
*/
if ( ! config - > readdir_attr_max_access ) {
attr_data - > attr_data . aapl . max_access = FILE_GENERIC_ALL ;
} else {
2021-06-08 21:56:25 +03:00
status = smbd_calculate_access_mask_fsp ( fsp - > conn - > cwd_fsp ,
fsp ,
2021-05-13 14:08:20 +03:00
false ,
SEC_FLAG_MAXIMUM_ALLOWED ,
& attr_data - > attr_data . aapl . max_access ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
goto fail ;
}
}
return NT_STATUS_OK ;
fail :
DBG_WARNING ( " Path [%s], error: %s \n " , fsp_str_dbg ( fsp ) ,
nt_errstr ( status ) ) ;
TALLOC_FREE ( * pattr_data ) ;
return status ;
}
2014-11-26 20:11:17 +03:00
static NTSTATUS fruit_fget_nt_acl ( vfs_handle_struct * handle ,
files_struct * fsp ,
2015-05-03 06:11:02 +03:00
uint32_t security_info ,
2014-11-26 20:11:17 +03:00
TALLOC_CTX * mem_ctx ,
struct security_descriptor * * ppdesc )
{
NTSTATUS status ;
struct security_ace ace ;
struct dom_sid sid ;
struct fruit_config_data * config ;
SMB_VFS_HANDLE_GET_DATA ( handle , config ,
struct fruit_config_data ,
return NT_STATUS_UNSUCCESSFUL ) ;
status = SMB_VFS_NEXT_FGET_NT_ACL ( handle , fsp , security_info ,
mem_ctx , ppdesc ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
}
/*
* Add MS NFS style ACEs with uid , gid and mode
*/
2017-07-12 10:33:59 +03:00
if ( ! global_fruit_config . nego_aapl ) {
return NT_STATUS_OK ;
}
2014-11-26 20:11:17 +03:00
if ( ! config - > unix_info_enabled ) {
return NT_STATUS_OK ;
}
2018-03-15 19:57:09 +03:00
/* First remove any existing ACE's with NFS style mode/uid/gid SIDs. */
status = remove_virtual_nfs_aces ( * ppdesc ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DBG_WARNING ( " failed to remove MS NFS style ACEs \n " ) ;
return status ;
}
2014-11-26 20:11:17 +03:00
/* MS NFS style mode */
sid_compose ( & sid , & global_sid_Unix_NFS_Mode , fsp - > fsp_name - > st . st_ex_mode ) ;
init_sec_ace ( & ace , & sid , SEC_ACE_TYPE_ACCESS_DENIED , 0 , 0 ) ;
status = security_descriptor_dacl_add ( * ppdesc , & ace ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 1 , ( " failed to add MS NFS style ACE \n " ) ) ;
return status ;
}
/* MS NFS style uid */
sid_compose ( & sid , & global_sid_Unix_NFS_Users , fsp - > fsp_name - > st . st_ex_uid ) ;
init_sec_ace ( & ace , & sid , SEC_ACE_TYPE_ACCESS_DENIED , 0 , 0 ) ;
status = security_descriptor_dacl_add ( * ppdesc , & ace ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 1 , ( " failed to add MS NFS style ACE \n " ) ) ;
return status ;
}
/* MS NFS style gid */
sid_compose ( & sid , & global_sid_Unix_NFS_Groups , fsp - > fsp_name - > st . st_ex_gid ) ;
init_sec_ace ( & ace , & sid , SEC_ACE_TYPE_ACCESS_DENIED , 0 , 0 ) ;
status = security_descriptor_dacl_add ( * ppdesc , & ace ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 1 , ( " failed to add MS NFS style ACE \n " ) ) ;
return status ;
}
return NT_STATUS_OK ;
}
static NTSTATUS fruit_fset_nt_acl ( vfs_handle_struct * handle ,
files_struct * fsp ,
2015-05-03 06:11:02 +03:00
uint32_t security_info_sent ,
2018-03-03 00:21:37 +03:00
const struct security_descriptor * orig_psd )
2014-11-26 20:11:17 +03:00
{
NTSTATUS status ;
bool do_chmod ;
2016-06-16 04:01:23 +03:00
mode_t ms_nfs_mode = 0 ;
2014-11-26 20:11:17 +03:00
int result ;
2018-03-03 00:21:37 +03:00
struct security_descriptor * psd = NULL ;
2018-03-03 00:51:54 +03:00
uint32_t orig_num_aces = 0 ;
if ( orig_psd - > dacl ! = NULL ) {
orig_num_aces = orig_psd - > dacl - > num_aces ;
}
2018-03-03 00:21:37 +03:00
psd = security_descriptor_copy ( talloc_tos ( ) , orig_psd ) ;
if ( psd = = NULL ) {
return NT_STATUS_NO_MEMORY ;
}
2014-11-26 20:11:17 +03:00
2024-07-29 13:39:32 +03:00
DBG_DEBUG ( " %s \n " , fsp_str_dbg ( fsp ) ) ;
2014-11-26 20:11:17 +03:00
status = check_ms_nfs ( handle , fsp , psd , & ms_nfs_mode , & do_chmod ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 1 , ( " fruit_fset_nt_acl: check_ms_nfs failed%s \n " , fsp_str_dbg ( fsp ) ) ) ;
2018-03-03 00:21:37 +03:00
TALLOC_FREE ( psd ) ;
2014-11-26 20:11:17 +03:00
return status ;
}
2018-03-03 00:51:54 +03:00
/*
* If only ms_nfs ACE entries were sent , ensure we set the DACL
* sent / present flags correctly now we ' ve removed them .
*/
if ( orig_num_aces ! = 0 ) {
/*
* Are there any ACE ' s left ?
*/
if ( psd - > dacl - > num_aces = = 0 ) {
/* No - clear the DACL sent/present flags. */
security_info_sent & = ~ SECINFO_DACL ;
psd - > type & = ~ SEC_DESC_DACL_PRESENT ;
}
}
2014-11-26 20:11:17 +03:00
status = SMB_VFS_NEXT_FSET_NT_ACL ( handle , fsp , security_info_sent , psd ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 1 , ( " fruit_fset_nt_acl: SMB_VFS_NEXT_FSET_NT_ACL failed%s \n " , fsp_str_dbg ( fsp ) ) ) ;
2018-03-03 00:21:37 +03:00
TALLOC_FREE ( psd ) ;
2014-11-26 20:11:17 +03:00
return status ;
}
if ( do_chmod ) {
2019-06-20 21:42:42 +03:00
result = SMB_VFS_FCHMOD ( fsp , ms_nfs_mode ) ;
2014-11-26 20:11:17 +03:00
if ( result ! = 0 ) {
2019-06-20 21:42:42 +03:00
DBG_WARNING ( " %s, result: %d, %04o error %s \n " ,
fsp_str_dbg ( fsp ) ,
result ,
( unsigned ) ms_nfs_mode ,
strerror ( errno ) ) ;
2014-11-26 20:11:17 +03:00
status = map_nt_error_from_unix ( errno ) ;
2018-03-03 00:21:37 +03:00
TALLOC_FREE ( psd ) ;
2014-11-26 20:11:17 +03:00
return status ;
}
}
2018-03-03 00:21:37 +03:00
TALLOC_FREE ( psd ) ;
2014-11-26 20:11:17 +03:00
return NT_STATUS_OK ;
}
2017-06-03 13:57:59 +03:00
static struct vfs_offload_ctx * fruit_offload_ctx ;
struct fruit_offload_read_state {
struct vfs_handle_struct * handle ;
struct tevent_context * ev ;
files_struct * fsp ;
uint32_t fsctl ;
2021-06-22 21:13:02 +03:00
uint32_t flags ;
uint64_t xferlen ;
2017-06-03 13:57:59 +03:00
DATA_BLOB token ;
} ;
static void fruit_offload_read_done ( struct tevent_req * subreq ) ;
static struct tevent_req * fruit_offload_read_send (
TALLOC_CTX * mem_ctx ,
struct tevent_context * ev ,
struct vfs_handle_struct * handle ,
files_struct * fsp ,
uint32_t fsctl ,
uint32_t ttl ,
off_t offset ,
size_t to_copy )
{
struct tevent_req * req = NULL ;
struct tevent_req * subreq = NULL ;
struct fruit_offload_read_state * state = NULL ;
req = tevent_req_create ( mem_ctx , & state ,
struct fruit_offload_read_state ) ;
if ( req = = NULL ) {
return NULL ;
}
* state = ( struct fruit_offload_read_state ) {
. handle = handle ,
. ev = ev ,
. fsp = fsp ,
. fsctl = fsctl ,
} ;
subreq = SMB_VFS_NEXT_OFFLOAD_READ_SEND ( mem_ctx , ev , handle , fsp ,
fsctl , ttl , offset , to_copy ) ;
if ( tevent_req_nomem ( subreq , req ) ) {
return tevent_req_post ( req , ev ) ;
}
tevent_req_set_callback ( subreq , fruit_offload_read_done , req ) ;
return req ;
}
static void fruit_offload_read_done ( struct tevent_req * subreq )
{
struct tevent_req * req = tevent_req_callback_data (
subreq , struct tevent_req ) ;
struct fruit_offload_read_state * state = tevent_req_data (
req , struct fruit_offload_read_state ) ;
NTSTATUS status ;
status = SMB_VFS_NEXT_OFFLOAD_READ_RECV ( subreq ,
state - > handle ,
state ,
2021-06-22 21:13:02 +03:00
& state - > flags ,
& state - > xferlen ,
2017-06-03 13:57:59 +03:00
& state - > token ) ;
TALLOC_FREE ( subreq ) ;
if ( tevent_req_nterror ( req , status ) ) {
return ;
}
if ( state - > fsctl ! = FSCTL_SRV_REQUEST_RESUME_KEY ) {
tevent_req_done ( req ) ;
return ;
}
status = vfs_offload_token_ctx_init ( state - > fsp - > conn - > sconn - > client ,
& fruit_offload_ctx ) ;
if ( tevent_req_nterror ( req , status ) ) {
return ;
}
status = vfs_offload_token_db_store_fsp ( fruit_offload_ctx ,
state - > fsp ,
& state - > token ) ;
if ( tevent_req_nterror ( req , status ) ) {
return ;
}
tevent_req_done ( req ) ;
return ;
}
static NTSTATUS fruit_offload_read_recv ( struct tevent_req * req ,
struct vfs_handle_struct * handle ,
TALLOC_CTX * mem_ctx ,
2021-06-22 21:13:02 +03:00
uint32_t * flags ,
uint64_t * xferlen ,
2017-06-03 13:57:59 +03:00
DATA_BLOB * token )
{
struct fruit_offload_read_state * state = tevent_req_data (
req , struct fruit_offload_read_state ) ;
NTSTATUS status ;
if ( tevent_req_is_nterror ( req , & status ) ) {
tevent_req_received ( req ) ;
return status ;
}
2021-06-22 21:13:02 +03:00
* flags = state - > flags ;
* xferlen = state - > xferlen ;
2017-06-03 13:57:59 +03:00
token - > length = state - > token . length ;
token - > data = talloc_move ( mem_ctx , & state - > token . data ) ;
tevent_req_received ( req ) ;
return NT_STATUS_OK ;
}
2017-06-04 14:50:33 +03:00
struct fruit_offload_write_state {
2015-04-22 23:29:16 +03:00
struct vfs_handle_struct * handle ;
off_t copied ;
struct files_struct * src_fsp ;
struct files_struct * dst_fsp ;
bool is_copyfile ;
} ;
2017-06-04 14:50:33 +03:00
static void fruit_offload_write_done ( struct tevent_req * subreq ) ;
static struct tevent_req * fruit_offload_write_send ( struct vfs_handle_struct * handle ,
2015-04-22 23:29:16 +03:00
TALLOC_CTX * mem_ctx ,
struct tevent_context * ev ,
2017-06-09 14:02:49 +03:00
uint32_t fsctl ,
DATA_BLOB * token ,
off_t transfer_offset ,
2015-04-22 23:29:16 +03:00
struct files_struct * dest_fsp ,
off_t dest_off ,
2017-06-10 10:05:55 +03:00
off_t num )
2015-04-22 23:29:16 +03:00
{
struct tevent_req * req , * subreq ;
2017-06-04 14:50:33 +03:00
struct fruit_offload_write_state * state ;
2015-04-22 23:29:16 +03:00
NTSTATUS status ;
struct fruit_config_data * config ;
2017-06-09 14:02:49 +03:00
off_t src_off = transfer_offset ;
files_struct * src_fsp = NULL ;
2015-04-22 23:29:16 +03:00
off_t to_copy = num ;
2017-06-09 18:27:17 +03:00
bool copyfile_enabled = false ;
2015-04-22 23:29:16 +03:00
2015-07-11 14:44:05 +03:00
DEBUG ( 10 , ( " soff: %ju, doff: %ju, len: %ju \n " ,
( uintmax_t ) src_off , ( uintmax_t ) dest_off , ( uintmax_t ) num ) ) ;
2015-04-22 23:29:16 +03:00
SMB_VFS_HANDLE_GET_DATA ( handle , config ,
struct fruit_config_data ,
return NULL ) ;
2017-06-04 14:50:33 +03:00
req = tevent_req_create ( mem_ctx , & state ,
struct fruit_offload_write_state ) ;
2015-04-22 23:29:16 +03:00
if ( req = = NULL ) {
return NULL ;
}
2017-06-04 14:50:33 +03:00
state - > handle = handle ;
state - > dst_fsp = dest_fsp ;
2015-04-22 23:29:16 +03:00
2017-06-09 14:02:49 +03:00
switch ( fsctl ) {
case FSCTL_SRV_COPYCHUNK :
case FSCTL_SRV_COPYCHUNK_WRITE :
2017-06-09 18:27:17 +03:00
copyfile_enabled = config - > copyfile_enabled ;
2017-06-09 14:02:49 +03:00
break ;
default :
break ;
}
2015-04-22 23:29:16 +03:00
/*
* Check if this a OS X copyfile style copychunk request with
* a requested chunk count of 0 that was translated to a
2017-06-04 14:50:33 +03:00
* offload_write_send VFS call overloading the parameters src_off
2015-04-22 23:29:16 +03:00
* = dest_off = num = 0.
*/
2017-06-09 18:27:17 +03:00
if ( copyfile_enabled & & num = = 0 & & src_off = = 0 & & dest_off = = 0 ) {
status = vfs_offload_token_db_fetch_fsp (
fruit_offload_ctx , token , & src_fsp ) ;
if ( tevent_req_nterror ( req , status ) ) {
return tevent_req_post ( req , ev ) ;
}
state - > src_fsp = src_fsp ;
2015-04-22 23:29:16 +03:00
status = vfs_stat_fsp ( src_fsp ) ;
if ( tevent_req_nterror ( req , status ) ) {
return tevent_req_post ( req , ev ) ;
}
to_copy = src_fsp - > fsp_name - > st . st_ex_size ;
2017-06-04 14:50:33 +03:00
state - > is_copyfile = true ;
2015-04-22 23:29:16 +03:00
}
2017-06-04 14:50:33 +03:00
subreq = SMB_VFS_NEXT_OFFLOAD_WRITE_SEND ( handle ,
2015-04-22 23:29:16 +03:00
mem_ctx ,
ev ,
2017-06-09 14:02:49 +03:00
fsctl ,
token ,
transfer_offset ,
2015-04-22 23:29:16 +03:00
dest_fsp ,
dest_off ,
2017-06-10 10:05:55 +03:00
to_copy ) ;
2015-04-22 23:29:16 +03:00
if ( tevent_req_nomem ( subreq , req ) ) {
return tevent_req_post ( req , ev ) ;
}
2017-06-04 14:50:33 +03:00
tevent_req_set_callback ( subreq , fruit_offload_write_done , req ) ;
2015-04-22 23:29:16 +03:00
return req ;
}
2017-06-04 14:50:33 +03:00
static void fruit_offload_write_done ( struct tevent_req * subreq )
2015-04-22 23:29:16 +03:00
{
struct tevent_req * req = tevent_req_callback_data (
subreq , struct tevent_req ) ;
2017-06-04 14:50:33 +03:00
struct fruit_offload_write_state * state = tevent_req_data (
req , struct fruit_offload_write_state ) ;
2015-04-22 23:29:16 +03:00
NTSTATUS status ;
unsigned int num_streams = 0 ;
struct stream_struct * streams = NULL ;
2016-05-19 19:42:04 +03:00
unsigned int i ;
2015-04-22 23:29:16 +03:00
struct smb_filename * src_fname_tmp = NULL ;
struct smb_filename * dst_fname_tmp = NULL ;
2017-06-04 14:50:33 +03:00
status = SMB_VFS_NEXT_OFFLOAD_WRITE_RECV ( state - > handle ,
2015-04-22 23:29:16 +03:00
subreq ,
& state - > copied ) ;
TALLOC_FREE ( subreq ) ;
if ( tevent_req_nterror ( req , status ) ) {
return ;
}
if ( ! state - > is_copyfile ) {
tevent_req_done ( req ) ;
return ;
}
/*
2017-01-20 01:49:54 +03:00
* Now copy all remaining streams . We know the share supports
2015-04-22 23:29:16 +03:00
* streams , because we ' re in vfs_fruit . We don ' t do this async
* because streams are few and small .
*/
2021-04-28 18:58:35 +03:00
status = vfs_fstreaminfo ( state - > src_fsp ,
2015-04-22 23:29:16 +03:00
req , & num_streams , & streams ) ;
if ( tevent_req_nterror ( req , status ) ) {
return ;
}
if ( num_streams = = 1 ) {
/* There is always one stream, ::$DATA. */
tevent_req_done ( req ) ;
return ;
}
for ( i = 0 ; i < num_streams ; i + + ) {
2015-07-01 02:43:09 +03:00
DEBUG ( 10 , ( " %s: stream: '%s'/%zu \n " ,
__func__ , streams [ i ] . name , ( size_t ) streams [ i ] . size ) ) ;
2015-04-22 23:29:16 +03:00
src_fname_tmp = synthetic_smb_fname (
req ,
state - > src_fsp - > fsp_name - > base_name ,
streams [ i ] . name ,
2016-03-19 07:19:38 +03:00
NULL ,
2020-04-30 12:48:32 +03:00
state - > src_fsp - > fsp_name - > twrp ,
2016-03-19 07:19:38 +03:00
state - > src_fsp - > fsp_name - > flags ) ;
2015-04-22 23:29:16 +03:00
if ( tevent_req_nomem ( src_fname_tmp , req ) ) {
return ;
}
if ( is_ntfs_default_stream_smb_fname ( src_fname_tmp ) ) {
TALLOC_FREE ( src_fname_tmp ) ;
continue ;
}
dst_fname_tmp = synthetic_smb_fname (
req ,
state - > dst_fsp - > fsp_name - > base_name ,
streams [ i ] . name ,
2016-03-19 07:19:38 +03:00
NULL ,
2020-04-30 12:48:32 +03:00
state - > dst_fsp - > fsp_name - > twrp ,
2016-03-19 07:19:38 +03:00
state - > dst_fsp - > fsp_name - > flags ) ;
2015-04-22 23:29:16 +03:00
if ( tevent_req_nomem ( dst_fname_tmp , req ) ) {
TALLOC_FREE ( src_fname_tmp ) ;
return ;
}
status = copy_file ( req ,
state - > handle - > conn ,
src_fname_tmp ,
dst_fname_tmp ,
2022-05-20 09:06:28 +03:00
FILE_CREATE ) ;
2015-04-22 23:29:16 +03:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 1 , ( " %s: copy %s to %s failed: %s \n " , __func__ ,
smb_fname_str_dbg ( src_fname_tmp ) ,
smb_fname_str_dbg ( dst_fname_tmp ) ,
nt_errstr ( status ) ) ) ;
TALLOC_FREE ( src_fname_tmp ) ;
TALLOC_FREE ( dst_fname_tmp ) ;
tevent_req_nterror ( req , status ) ;
return ;
}
TALLOC_FREE ( src_fname_tmp ) ;
TALLOC_FREE ( dst_fname_tmp ) ;
}
TALLOC_FREE ( streams ) ;
TALLOC_FREE ( src_fname_tmp ) ;
TALLOC_FREE ( dst_fname_tmp ) ;
tevent_req_done ( req ) ;
}
2017-06-04 14:50:33 +03:00
static NTSTATUS fruit_offload_write_recv ( struct vfs_handle_struct * handle ,
2015-04-22 23:29:16 +03:00
struct tevent_req * req ,
off_t * copied )
{
2017-06-04 14:50:33 +03:00
struct fruit_offload_write_state * state = tevent_req_data (
req , struct fruit_offload_write_state ) ;
2015-04-22 23:29:16 +03:00
NTSTATUS status ;
if ( tevent_req_is_nterror ( req , & status ) ) {
DEBUG ( 1 , ( " server side copy chunk failed: %s \n " ,
nt_errstr ( status ) ) ) ;
* copied = 0 ;
tevent_req_received ( req ) ;
return status ;
}
2017-06-04 14:50:33 +03:00
* copied = state - > copied ;
2015-04-22 23:29:16 +03:00
tevent_req_received ( req ) ;
return NT_STATUS_OK ;
}
2017-11-03 12:56:29 +03:00
static char * fruit_get_bandsize_line ( char * * lines , int numlines )
{
static regex_t re ;
static bool re_initialized = false ;
int i ;
int ret ;
if ( ! re_initialized ) {
ret = regcomp ( & re , " ^[[:blank:]]*<key>band-size</key>$ " , 0 ) ;
if ( ret ! = 0 ) {
return NULL ;
}
re_initialized = true ;
}
for ( i = 0 ; i < numlines ; i + + ) {
regmatch_t matches [ 1 ] ;
ret = regexec ( & re , lines [ i ] , 1 , matches , 0 ) ;
if ( ret = = 0 ) {
/*
* Check if the match was on the last line , sa we want
* the subsequent line .
*/
if ( i + 1 = = numlines ) {
return NULL ;
}
return lines [ i + 1 ] ;
}
if ( ret ! = REG_NOMATCH ) {
return NULL ;
}
}
return NULL ;
}
static bool fruit_get_bandsize_from_line ( char * line , size_t * _band_size )
{
static regex_t re ;
static bool re_initialized = false ;
regmatch_t matches [ 2 ] ;
uint64_t band_size ;
int ret ;
bool ok ;
if ( ! re_initialized ) {
ret = regcomp ( & re ,
" ^[[:blank:]]* "
" <integer> \\ ([[:digit:]]* \\ )</integer>$ " ,
0 ) ;
if ( ret ! = 0 ) {
return false ;
}
re_initialized = true ;
}
ret = regexec ( & re , line , 2 , matches , 0 ) ;
if ( ret ! = 0 ) {
DBG_ERR ( " regex failed [%s] \n " , line ) ;
return false ;
}
line [ matches [ 1 ] . rm_eo ] = ' \0 ' ;
ok = conv_str_u64 ( & line [ matches [ 1 ] . rm_so ] , & band_size ) ;
if ( ! ok ) {
return false ;
}
* _band_size = ( size_t ) band_size ;
return true ;
}
/*
* This reads and parses an Info . plist from a TM sparsebundle looking for the
* " band-size " key and value .
*/
static bool fruit_get_bandsize ( vfs_handle_struct * handle ,
const char * dir ,
size_t * band_size )
{
# define INFO_PLIST_MAX_SIZE 64*1024
char * plist = NULL ;
struct smb_filename * smb_fname = NULL ;
files_struct * fsp = NULL ;
uint8_t * file_data = NULL ;
char * * lines = NULL ;
char * band_size_line = NULL ;
size_t plist_file_size ;
ssize_t nread ;
int numlines ;
int ret ;
bool ok = false ;
NTSTATUS status ;
plist = talloc_asprintf ( talloc_tos ( ) ,
" %s/%s/Info.plist " ,
handle - > conn - > connectpath ,
dir ) ;
if ( plist = = NULL ) {
ok = false ;
goto out ;
}
2020-05-03 16:04:14 +03:00
smb_fname = synthetic_smb_fname ( talloc_tos ( ) ,
plist ,
NULL ,
NULL ,
2020-04-30 12:48:32 +03:00
0 ,
2020-05-03 16:04:14 +03:00
0 ) ;
2017-11-03 12:56:29 +03:00
if ( smb_fname = = NULL ) {
ok = false ;
goto out ;
}
ret = SMB_VFS_NEXT_LSTAT ( handle , smb_fname ) ;
if ( ret ! = 0 ) {
DBG_INFO ( " Ignoring Sparsebundle without Info.plist [%s] \n " , dir ) ;
ok = true ;
goto out ;
}
plist_file_size = smb_fname - > st . st_ex_size ;
if ( plist_file_size > INFO_PLIST_MAX_SIZE ) {
DBG_INFO ( " %s is too large, ignoring \n " , plist ) ;
ok = true ;
goto out ;
}
status = SMB_VFS_NEXT_CREATE_FILE (
handle , /* conn */
NULL , /* req */
2021-11-23 14:29:17 +03:00
NULL , /* dirfsp */
2017-11-03 12:56:29 +03:00
smb_fname , /* fname */
FILE_GENERIC_READ , /* access_mask */
FILE_SHARE_READ | FILE_SHARE_WRITE , /* share_access */
FILE_OPEN , /* create_disposition */
0 , /* create_options */
0 , /* file_attributes */
INTERNAL_OPEN_ONLY , /* oplock_request */
NULL , /* lease */
0 , /* allocation_size */
0 , /* private_flags */
NULL , /* sd */
NULL , /* ea_list */
& fsp , /* result */
NULL , /* psbuf */
NULL , NULL ) ; /* create context */
if ( ! NT_STATUS_IS_OK ( status ) ) {
DBG_INFO ( " Opening [%s] failed [%s] \n " ,
smb_fname_str_dbg ( smb_fname ) , nt_errstr ( status ) ) ;
ok = false ;
goto out ;
}
2020-08-06 13:55:33 +03:00
file_data = talloc_zero_array ( talloc_tos ( ) ,
uint8_t ,
plist_file_size + 1 ) ;
2017-11-03 12:56:29 +03:00
if ( file_data = = NULL ) {
ok = false ;
goto out ;
}
nread = SMB_VFS_NEXT_PREAD ( handle , fsp , file_data , plist_file_size , 0 ) ;
if ( nread ! = plist_file_size ) {
DBG_ERR ( " Short read on [%s]: %zu/%zd \n " ,
fsp_str_dbg ( fsp ) , nread , plist_file_size ) ;
ok = false ;
goto out ;
}
2022-02-01 19:47:29 +03:00
status = close_file_free ( NULL , & fsp , NORMAL_CLOSE ) ;
2017-11-03 12:56:29 +03:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
DBG_ERR ( " close_file failed: %s \n " , nt_errstr ( status ) ) ;
ok = false ;
goto out ;
}
lines = file_lines_parse ( ( char * ) file_data ,
plist_file_size ,
& numlines ,
talloc_tos ( ) ) ;
if ( lines = = NULL ) {
ok = false ;
goto out ;
}
band_size_line = fruit_get_bandsize_line ( lines , numlines ) ;
if ( band_size_line = = NULL ) {
DBG_ERR ( " Didn't find band-size key in [%s] \n " ,
smb_fname_str_dbg ( smb_fname ) ) ;
ok = false ;
goto out ;
}
ok = fruit_get_bandsize_from_line ( band_size_line , band_size ) ;
if ( ! ok ) {
DBG_ERR ( " fruit_get_bandsize_from_line failed \n " ) ;
goto out ;
}
DBG_DEBUG ( " Parsed band-size [%zu] for [%s] \n " , * band_size , plist ) ;
out :
if ( fsp ! = NULL ) {
2022-02-01 19:47:29 +03:00
status = close_file_free ( NULL , & fsp , NORMAL_CLOSE ) ;
2017-11-03 12:56:29 +03:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
DBG_ERR ( " close_file failed: %s \n " , nt_errstr ( status ) ) ;
}
}
TALLOC_FREE ( plist ) ;
TALLOC_FREE ( smb_fname ) ;
TALLOC_FREE ( file_data ) ;
TALLOC_FREE ( lines ) ;
return ok ;
}
struct fruit_disk_free_state {
2018-02-22 17:52:46 +03:00
off_t total_size ;
2017-11-03 12:56:29 +03:00
} ;
static bool fruit_get_num_bands ( vfs_handle_struct * handle ,
2020-03-19 14:39:44 +03:00
const char * bundle ,
2017-11-03 12:56:29 +03:00
size_t * _nbands )
{
char * path = NULL ;
struct smb_filename * bands_dir = NULL ;
2020-03-19 14:03:27 +03:00
struct smb_Dir * dir_hnd = NULL ;
const char * dname = NULL ;
char * talloced = NULL ;
2017-11-03 12:56:29 +03:00
size_t nbands ;
2022-03-01 01:08:40 +03:00
NTSTATUS status ;
2017-11-03 12:56:29 +03:00
path = talloc_asprintf ( talloc_tos ( ) ,
" %s/%s/bands " ,
handle - > conn - > connectpath ,
bundle ) ;
if ( path = = NULL ) {
return false ;
}
bands_dir = synthetic_smb_fname ( talloc_tos ( ) ,
path ,
NULL ,
NULL ,
2020-04-30 12:48:32 +03:00
0 ,
2017-11-03 12:56:29 +03:00
0 ) ;
TALLOC_FREE ( path ) ;
if ( bands_dir = = NULL ) {
return false ;
}
2022-03-01 01:34:48 +03:00
status = OpenDir ( talloc_tos ( ) ,
handle - > conn ,
bands_dir ,
NULL ,
0 ,
& dir_hnd ) ;
2022-03-01 01:08:40 +03:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
2017-11-03 12:56:29 +03:00
TALLOC_FREE ( bands_dir ) ;
2022-03-01 01:08:40 +03:00
errno = map_errno_from_nt_status ( status ) ;
2017-11-03 12:56:29 +03:00
return false ;
}
nbands = 0 ;
2023-06-20 13:25:45 +03:00
while ( ( dname = ReadDirName ( dir_hnd , & talloced ) ) ! = NULL ) {
2020-03-19 14:03:27 +03:00
if ( ISDOT ( dname ) | | ISDOTDOT ( dname ) ) {
2017-11-03 12:56:29 +03:00
continue ;
}
nbands + + ;
}
2020-03-19 14:03:27 +03:00
TALLOC_FREE ( dir_hnd ) ;
2017-11-03 12:56:29 +03:00
DBG_DEBUG ( " %zu bands in [%s] \n " , nbands , smb_fname_str_dbg ( bands_dir ) ) ;
TALLOC_FREE ( bands_dir ) ;
* _nbands = nbands ;
return true ;
}
static bool fruit_tmsize_do_dirent ( vfs_handle_struct * handle ,
struct fruit_disk_free_state * state ,
2020-03-19 14:42:45 +03:00
const char * name )
2017-11-03 12:56:29 +03:00
{
bool ok ;
char * p = NULL ;
size_t sparsebundle_strlen = strlen ( " sparsebundle " ) ;
2018-01-09 14:08:01 +03:00
size_t bandsize = 0 ;
2017-11-03 12:56:29 +03:00
size_t nbands ;
2018-02-22 17:52:46 +03:00
off_t tm_size ;
2017-11-03 12:56:29 +03:00
2020-03-19 14:42:45 +03:00
p = strstr ( name , " sparsebundle " ) ;
2017-11-03 12:56:29 +03:00
if ( p = = NULL ) {
return true ;
}
if ( p [ sparsebundle_strlen ] ! = ' \0 ' ) {
return true ;
}
2020-03-19 14:42:45 +03:00
DBG_DEBUG ( " Processing sparsebundle [%s] \n " , name ) ;
2017-11-03 12:56:29 +03:00
2020-03-19 14:42:45 +03:00
ok = fruit_get_bandsize ( handle , name , & bandsize ) ;
2017-11-03 12:56:29 +03:00
if ( ! ok ) {
/*
* Beware of race conditions : this may be an uninitialized
* Info . plist that a client is just creating . We don ' t want let
* this to trigger complete failure .
*/
2020-03-19 14:42:45 +03:00
DBG_ERR ( " Processing sparsebundle [%s] failed \n " , name ) ;
2017-11-03 12:56:29 +03:00
return true ;
}
2020-03-19 14:42:45 +03:00
ok = fruit_get_num_bands ( handle , name , & nbands ) ;
2017-11-03 12:56:29 +03:00
if ( ! ok ) {
/*
* Beware of race conditions : this may be a backup sparsebundle
* in an early stage lacking a bands subdirectory . We don ' t want
* let this to trigger complete failure .
*/
2020-03-19 14:42:45 +03:00
DBG_ERR ( " Processing sparsebundle [%s] failed \n " , name ) ;
2017-11-03 12:56:29 +03:00
return true ;
}
2020-03-04 00:51:46 +03:00
/*
* Arithmetic on 32 - bit systems may cause overflow , depending on
* size_t precision . First we check its unlikely , then we
* force the precision into target off_t , then we check that
* the total did not overflow either .
*/
2018-03-28 15:17:59 +03:00
if ( bandsize > SIZE_MAX / nbands ) {
2020-03-04 00:51:46 +03:00
DBG_ERR ( " tmsize potential overflow: bandsize [%zu] nbands [%zu] \n " ,
2017-11-03 12:56:29 +03:00
bandsize , nbands ) ;
return false ;
}
2020-03-04 00:51:46 +03:00
tm_size = ( off_t ) bandsize * ( off_t ) nbands ;
2017-11-03 12:56:29 +03:00
if ( state - > total_size + tm_size < state - > total_size ) {
2020-03-04 00:51:46 +03:00
DBG_ERR ( " tm total size overflow: bandsize [%zu] nbands [%zu] \n " ,
2017-11-03 12:56:29 +03:00
bandsize , nbands ) ;
return false ;
}
state - > total_size + = tm_size ;
2018-02-22 17:52:46 +03:00
DBG_DEBUG ( " [%s] tm_size [%jd] total_size [%jd] \n " ,
2020-03-19 14:42:45 +03:00
name , ( intmax_t ) tm_size , ( intmax_t ) state - > total_size ) ;
2017-11-03 12:56:29 +03:00
return true ;
}
/**
* Calculate used size of a TimeMachine volume
*
* This assumes that the volume is used only for TimeMachine .
*
* - readdir ( basedir of share ) , then
* - for every element that matches regex " ^ \ (.* \ ) \ .sparsebundle$ " :
* - parse " \1 .sparsebundle/Info.plist " and read the band - size XML key
* - count band files in " \1 .sparsebundle/bands/ "
* - calculate used size of all bands : band_count * band_size
* */
static uint64_t fruit_disk_free ( vfs_handle_struct * handle ,
const struct smb_filename * smb_fname ,
uint64_t * _bsize ,
uint64_t * _dfree ,
uint64_t * _dsize )
{
struct fruit_config_data * config = NULL ;
struct fruit_disk_free_state state = { 0 } ;
2020-03-19 14:03:27 +03:00
struct smb_Dir * dir_hnd = NULL ;
const char * dname = NULL ;
char * talloced = NULL ;
2017-11-03 12:56:29 +03:00
uint64_t dfree ;
uint64_t dsize ;
bool ok ;
2022-03-01 01:08:40 +03:00
NTSTATUS status ;
2017-11-03 12:56:29 +03:00
SMB_VFS_HANDLE_GET_DATA ( handle , config ,
struct fruit_config_data ,
return UINT64_MAX ) ;
if ( ! config - > time_machine | |
config - > time_machine_max_size = = 0 )
{
return SMB_VFS_NEXT_DISK_FREE ( handle ,
smb_fname ,
_bsize ,
_dfree ,
_dsize ) ;
}
2022-03-01 01:34:48 +03:00
status = OpenDir ( talloc_tos ( ) ,
handle - > conn ,
smb_fname ,
NULL ,
0 ,
& dir_hnd ) ;
2022-03-01 01:08:40 +03:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
errno = map_errno_from_nt_status ( status ) ;
2017-11-03 12:56:29 +03:00
return UINT64_MAX ;
}
2023-06-20 13:25:45 +03:00
while ( ( dname = ReadDirName ( dir_hnd , & talloced ) ) ! = NULL ) {
2020-03-19 14:03:27 +03:00
ok = fruit_tmsize_do_dirent ( handle , & state , dname ) ;
2017-11-03 12:56:29 +03:00
if ( ! ok ) {
2020-03-19 14:03:27 +03:00
TALLOC_FREE ( talloced ) ;
TALLOC_FREE ( dir_hnd ) ;
2017-11-03 12:56:29 +03:00
return UINT64_MAX ;
}
2020-03-19 14:03:27 +03:00
TALLOC_FREE ( talloced ) ;
2017-11-03 12:56:29 +03:00
}
2020-03-19 14:03:27 +03:00
TALLOC_FREE ( dir_hnd ) ;
2017-11-03 12:56:29 +03:00
dsize = config - > time_machine_max_size / 512 ;
dfree = dsize - ( state . total_size / 512 ) ;
if ( dfree > dsize ) {
dfree = 0 ;
}
* _bsize = 512 ;
* _dsize = dsize ;
* _dfree = dfree ;
return dfree / 2 ;
}
2019-06-29 15:35:49 +03:00
static uint64_t fruit_fs_file_id ( struct vfs_handle_struct * handle ,
const SMB_STRUCT_STAT * psbuf )
{
struct fruit_config_data * config = NULL ;
SMB_VFS_HANDLE_GET_DATA ( handle , config ,
struct fruit_config_data ,
return 0 ) ;
2019-06-30 15:24:59 +03:00
if ( global_fruit_config . nego_aapl & &
config - > aapl_zero_file_id )
{
2019-06-29 15:35:49 +03:00
return 0 ;
}
return SMB_VFS_NEXT_FS_FILE_ID ( handle , psbuf ) ;
}
2014-06-23 18:59:45 +04:00
static struct vfs_fn_pointers vfs_fruit_fns = {
. connect_fn = fruit_connect ,
2017-11-03 12:56:29 +03:00
. disk_free_fn = fruit_disk_free ,
2014-06-23 18:59:45 +04:00
/* File operations */
2021-04-09 16:53:20 +03:00
. fchmod_fn = fruit_fchmod ,
2019-09-12 20:46:02 +03:00
. unlinkat_fn = fruit_unlinkat ,
2019-08-10 00:22:03 +03:00
. renameat_fn = fruit_renameat ,
2020-05-21 00:04:26 +03:00
. openat_fn = fruit_openat ,
2018-12-18 19:18:33 +03:00
. close_fn = fruit_close ,
2014-06-23 18:59:45 +04:00
. pread_fn = fruit_pread ,
. pwrite_fn = fruit_pwrite ,
2017-05-12 15:40:03 +03:00
. pread_send_fn = fruit_pread_send ,
. pread_recv_fn = fruit_pread_recv ,
. pwrite_send_fn = fruit_pwrite_send ,
. pwrite_recv_fn = fruit_pwrite_recv ,
2022-09-20 23:25:22 +03:00
. fsync_send_fn = fruit_fsync_send ,
. fsync_recv_fn = fruit_fsync_recv ,
2014-06-23 18:59:45 +04:00
. stat_fn = fruit_stat ,
. lstat_fn = fruit_lstat ,
. fstat_fn = fruit_fstat ,
2021-04-27 18:09:35 +03:00
. fstreaminfo_fn = fruit_fstreaminfo ,
2021-04-13 17:23:41 +03:00
. fntimes_fn = fruit_fntimes ,
2014-06-23 18:59:45 +04:00
. ftruncate_fn = fruit_ftruncate ,
. fallocate_fn = fruit_fallocate ,
. create_file_fn = fruit_create_file ,
2021-05-13 14:08:20 +03:00
. freaddir_attr_fn = fruit_freaddir_attr ,
2017-06-03 13:57:59 +03:00
. offload_read_send_fn = fruit_offload_read_send ,
. offload_read_recv_fn = fruit_offload_read_recv ,
2017-06-04 14:50:33 +03:00
. offload_write_send_fn = fruit_offload_write_send ,
. offload_write_recv_fn = fruit_offload_write_recv ,
2019-06-29 15:35:49 +03:00
. fs_file_id_fn = fruit_fs_file_id ,
2014-11-26 20:11:17 +03:00
/* NT ACL operations */
. fget_nt_acl_fn = fruit_fget_nt_acl ,
. fset_nt_acl_fn = fruit_fset_nt_acl ,
2014-06-23 18:59:45 +04:00
} ;
2017-12-16 01:32:12 +03:00
static_decl_vfs ;
2017-04-20 22:24:43 +03:00
NTSTATUS vfs_fruit_init ( TALLOC_CTX * ctx )
2014-06-23 18:59:45 +04:00
{
NTSTATUS ret = smb_register_vfs ( SMB_VFS_INTERFACE_VERSION , " fruit " ,
& vfs_fruit_fns ) ;
if ( ! NT_STATUS_IS_OK ( ret ) ) {
return ret ;
}
vfs_fruit_debug_level = debug_add_class ( " fruit " ) ;
if ( vfs_fruit_debug_level = = - 1 ) {
vfs_fruit_debug_level = DBGC_VFS ;
DEBUG ( 0 , ( " %s: Couldn't register custom debugging class! \n " ,
" vfs_fruit_init " ) ) ;
} else {
DEBUG ( 10 , ( " %s: Debug class number of '%s': %d \n " ,
" vfs_fruit_init " , " fruit " , vfs_fruit_debug_level ) ) ;
}
return ret ;
}