LCOV - code coverage report
Current view: top level - source3/modules - vfs_fruit.c (source / functions) Hit Total Coverage
Test: coverage report for fix-15632 9995c5c2 Lines: 1768 2343 75.5 %
Date: 2024-04-13 12:30:31 Functions: 116 128 90.6 %

          Line data    Source code
       1             : /*
       2             :  * OS X and Netatalk interoperability VFS module for Samba-3.x
       3             :  *
       4             :  * Copyright (C) Ralph Boehme, 2013, 2014
       5             :  *
       6             :  * This program is free software; you can redistribute it and/or modify
       7             :  * it under the terms of the GNU General Public License as published by
       8             :  * the Free Software Foundation; either version 3 of the License, or
       9             :  * (at your option) any later version.
      10             :  *
      11             :  * This program is distributed in the hope that it will be useful,
      12             :  * but WITHOUT ANY WARRANTY; without even the implied warranty of
      13             :  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
      14             :  * GNU General Public License for more details.
      15             :  *
      16             :  * You should have received a copy of the GNU General Public License
      17             :  * along with this program; if not, see <http://www.gnu.org/licenses/>.
      18             :  */
      19             : 
      20             : #include "includes.h"
      21             : #include "MacExtensions.h"
      22             : #include "smbd/smbd.h"
      23             : #include "system/filesys.h"
      24             : #include "lib/util/time.h"
      25             : #include "system/shmem.h"
      26             : #include "locking/proto.h"
      27             : #include "smbd/globals.h"
      28             : #include "messages.h"
      29             : #include "libcli/security/security.h"
      30             : #include "../libcli/smb/smb2_create_ctx.h"
      31             : #include "lib/util/tevent_ntstatus.h"
      32             : #include "lib/util/tevent_unix.h"
      33             : #include "offload_token.h"
      34             : #include "string_replace.h"
      35             : #include "hash_inode.h"
      36             : #include "lib/adouble.h"
      37             : #include "lib/util_macstreams.h"
      38             : #include "source3/smbd/dir.h"
      39             : 
      40             : /*
      41             :  * Enhanced OS X and Netatalk compatibility
      42             :  * ========================================
      43             :  *
      44             :  * This modules takes advantage of vfs_streams_xattr and
      45             :  * vfs_catia. VFS modules vfs_fruit and vfs_streams_xattr must be
      46             :  * loaded in the correct order:
      47             :  *
      48             :  *   vfs modules = catia fruit streams_xattr
      49             :  *
      50             :  * The module intercepts the OS X special streams "AFP_AfpInfo" and
      51             :  * "AFP_Resource" and handles them in a special way. All other named
      52             :  * streams are deferred to vfs_streams_xattr.
      53             :  *
      54             :  * The OS X client maps all NTFS illegal characters to the Unicode
      55             :  * private range. This module optionally stores the characters using
      56             :  * their native ASCII encoding using vfs_catia. If you're not enabling
      57             :  * this feature, you can skip catia from vfs modules.
      58             :  *
      59             :  * Finally, open modes are optionally checked against Netatalk AFP
      60             :  * share modes.
      61             :  *
      62             :  * The "AFP_AfpInfo" named stream is a binary blob containing OS X
      63             :  * extended metadata for files and directories. This module optionally
      64             :  * reads and stores this metadata in a way compatible with Netatalk 3
      65             :  * which stores the metadata in an EA "org.netatalk.metadata". Cf
      66             :  * source3/include/MacExtensions.h for a description of the binary
      67             :  * blobs content.
      68             :  *
      69             :  * The "AFP_Resource" named stream may be arbitrarily large, thus it
      70             :  * can't be stored in an xattr on most filesystem. ZFS on Solaris is
      71             :  * the only available filesystem where xattrs can be of any size and
      72             :  * the OS supports using the file APIs for xattrs.
      73             :  *
      74             :  * The AFP_Resource stream is stored in an AppleDouble file prepending
      75             :  * "._" to the filename. On Solaris with ZFS the stream is optionally
      76             :  * stored in an EA "org.netatalk.resource".
      77             :  *
      78             :  *
      79             :  * Extended Attributes
      80             :  * ===================
      81             :  *
      82             :  * The OS X SMB client sends xattrs as ADS too. For xattr interop with
      83             :  * other protocols you may want to adjust the xattr names the VFS
      84             :  * module vfs_streams_xattr uses for storing ADS's. This defaults to
      85             :  * user.DosStream.ADS_NAME:$DATA and can be changed by specifying
      86             :  * these module parameters:
      87             :  *
      88             :  *   streams_xattr:prefix = user.
      89             :  *   streams_xattr:store_stream_type = false
      90             :  *
      91             :  *
      92             :  * TODO
      93             :  * ====
      94             :  *
      95             :  * - log diagnostic if any needed VFS module is not loaded
      96             :  *   (eg with lp_vfs_objects())
      97             :  * - add tests
      98             :  */
      99             : 
     100             : static int vfs_fruit_debug_level = DBGC_VFS;
     101             : 
     102             : static struct global_fruit_config {
     103             :         bool nego_aapl; /* client negotiated AAPL */
     104             : 
     105             : } global_fruit_config;
     106             : 
     107             : #undef DBGC_CLASS
     108             : #define DBGC_CLASS vfs_fruit_debug_level
     109             : 
     110             : #define FRUIT_PARAM_TYPE_NAME "fruit"
     111             : 
     112             : enum apple_fork {APPLE_FORK_DATA, APPLE_FORK_RSRC};
     113             : 
     114             : enum fruit_rsrc {FRUIT_RSRC_STREAM, FRUIT_RSRC_ADFILE, FRUIT_RSRC_XATTR};
     115             : enum fruit_meta {FRUIT_META_STREAM, FRUIT_META_NETATALK};
     116             : enum fruit_locking {FRUIT_LOCKING_NETATALK, FRUIT_LOCKING_NONE};
     117             : enum fruit_encoding {FRUIT_ENC_NATIVE, FRUIT_ENC_PRIVATE};
     118             : 
     119             : struct fruit_config_data {
     120             :         enum fruit_rsrc rsrc;
     121             :         enum fruit_meta meta;
     122             :         enum fruit_locking locking;
     123             :         enum fruit_encoding encoding;
     124             :         bool use_aapl;          /* config from smb.conf */
     125             :         bool use_copyfile;
     126             :         bool readdir_attr_enabled;
     127             :         bool unix_info_enabled;
     128             :         bool copyfile_enabled;
     129             :         bool veto_appledouble;
     130             :         bool posix_rename;
     131             :         bool aapl_zero_file_id;
     132             :         const char *model;
     133             :         bool time_machine;
     134             :         off_t time_machine_max_size;
     135             :         bool convert_adouble;
     136             :         bool wipe_intentionally_left_blank_rfork;
     137             :         bool delete_empty_adfiles;
     138             :         bool validate_afpinfo;
     139             : 
     140             :         /*
     141             :          * Additional options, all enabled by default,
     142             :          * possibly useful for analyzing performance. The associated
     143             :          * operations with each of them may be expensive, so having
     144             :          * the chance to disable them individually gives a chance
     145             :          * tweaking the setup for the particular usecase.
     146             :          */
     147             :         bool readdir_attr_rsize;
     148             :         bool readdir_attr_finder_info;
     149             :         bool readdir_attr_max_access;
     150             :         /* Recursion guard. Will go away when we have STATX. */
     151             :         bool in_openat_pathref_fsp;
     152             : };
     153             : 
     154             : static const struct enum_list fruit_rsrc[] = {
     155             :         {FRUIT_RSRC_STREAM, "stream"}, /* pass on to vfs_streams_xattr */
     156             :         {FRUIT_RSRC_ADFILE, "file"}, /* ._ AppleDouble file */
     157             :         {FRUIT_RSRC_XATTR, "xattr"}, /* Netatalk compatible xattr (ZFS only) */
     158             :         { -1, NULL}
     159             : };
     160             : 
     161             : static const struct enum_list fruit_meta[] = {
     162             :         {FRUIT_META_STREAM, "stream"}, /* pass on to vfs_streams_xattr */
     163             :         {FRUIT_META_NETATALK, "netatalk"}, /* Netatalk compatible xattr */
     164             :         { -1, NULL}
     165             : };
     166             : 
     167             : static const struct enum_list fruit_locking[] = {
     168             :         {FRUIT_LOCKING_NETATALK, "netatalk"}, /* synchronize locks with Netatalk */
     169             :         {FRUIT_LOCKING_NONE, "none"},
     170             :         { -1, NULL}
     171             : };
     172             : 
     173             : static const struct enum_list fruit_encoding[] = {
     174             :         {FRUIT_ENC_NATIVE, "native"}, /* map unicode private chars to ASCII */
     175             :         {FRUIT_ENC_PRIVATE, "private"}, /* keep unicode private chars */
     176             :         { -1, NULL}
     177             : };
     178             : 
     179             : struct fio {
     180             :         vfs_handle_struct *handle;
     181             :         files_struct *fsp; /* backlink to itself */
     182             : 
     183             :         /* tcon config handle */
     184             :         struct fruit_config_data *config;
     185             : 
     186             :         /* Backend fsp for AppleDouble file, can be NULL */
     187             :         files_struct *ad_fsp;
     188             :         /* link from adouble_open_from_base_fsp() to fio */
     189             :         struct fio *real_fio;
     190             : 
     191             :         /* Denote stream type, meta or rsrc */
     192             :         adouble_type_t type;
     193             : 
     194             :         /*
     195             :          * AFP_AfpInfo stream created, but not written yet, thus still a fake
     196             :          * pipe fd. This is set to true in fruit_open_meta if there was no
     197             :          * existing stream but the caller requested O_CREAT. It is later set to
     198             :          * false when we get a write on the stream that then does open and
     199             :          * create the stream.
     200             :          */
     201             :         bool fake_fd;
     202             :         int flags;
     203             :         int mode;
     204             : };
     205             : 
     206             : /*****************************************************************************
     207             :  * Helper functions
     208             :  *****************************************************************************/
     209             : 
     210       64542 : static struct adouble *ad_get_meta_fsp(TALLOC_CTX *ctx,
     211             :                                        vfs_handle_struct *handle,
     212             :                                        const struct smb_filename *smb_fname)
     213             : {
     214             :         NTSTATUS status;
     215       64542 :         struct adouble *ad = NULL;
     216       64542 :         struct smb_filename *smb_fname_cp = NULL;
     217       64542 :         struct fruit_config_data *config = NULL;
     218             : 
     219       64542 :         if (smb_fname->fsp != NULL) {
     220        2946 :                 return ad_get(ctx, handle, smb_fname, ADOUBLE_META);
     221             :         }
     222             : 
     223       61596 :         SMB_VFS_HANDLE_GET_DATA(handle,
     224             :                                 config,
     225             :                                 struct fruit_config_data,
     226             :                                 return NULL);
     227             : 
     228       61596 :         if (config->in_openat_pathref_fsp) {
     229       25944 :                 return NULL;
     230             :         }
     231             : 
     232       35652 :         smb_fname_cp = cp_smb_filename(ctx,
     233             :                                        smb_fname);
     234       35652 :         if (smb_fname_cp == NULL) {
     235           0 :                 return NULL;
     236             :         }
     237       35652 :         TALLOC_FREE(smb_fname_cp->stream_name);
     238       35652 :         config->in_openat_pathref_fsp = true;
     239       35652 :         status = openat_pathref_fsp(handle->conn->cwd_fsp,
     240             :                                     smb_fname_cp);
     241       35652 :         config->in_openat_pathref_fsp = false;
     242       35652 :         if (!NT_STATUS_IS_OK(status)) {
     243       14926 :                 TALLOC_FREE(smb_fname_cp);
     244       14926 :                 return NULL;
     245             :         }
     246             : 
     247       20726 :         ad = ad_get(ctx, handle, smb_fname_cp, ADOUBLE_META);
     248       20726 :         TALLOC_FREE(smb_fname_cp);
     249       20726 :         return ad;
     250             : }
     251             : 
     252      113772 : static struct fio *fruit_get_complete_fio(vfs_handle_struct *handle,
     253             :                                           files_struct *fsp)
     254             : {
     255      113772 :         struct fio *fio = (struct fio *)VFS_FETCH_FSP_EXTENSION(handle, fsp);
     256             : 
     257      113772 :         if (fio == NULL) {
     258       91770 :                 return NULL;
     259             :         }
     260             : 
     261       22002 :         if (fio->real_fio != NULL) {
     262             :                 /*
     263             :                  * This is an fsp from adouble_open_from_base_fsp()
     264             :                  * we should just pass this to the next
     265             :                  * module.
     266             :                  */
     267          30 :                 return NULL;
     268             :         }
     269             : 
     270       21972 :         return fio;
     271             : }
     272             : 
     273             : /**
     274             :  * Initialize config struct from our smb.conf config parameters
     275             :  **/
     276         360 : static int init_fruit_config(vfs_handle_struct *handle)
     277             : {
     278             :         struct fruit_config_data *config;
     279         360 :         int enumval = -1;
     280         360 :         const char *tm_size_str = NULL;
     281             : 
     282         360 :         config = talloc_zero(handle->conn, struct fruit_config_data);
     283         360 :         if (!config) {
     284           0 :                 DEBUG(1, ("talloc_zero() failed\n"));
     285           0 :                 errno = ENOMEM;
     286           0 :                 return -1;
     287             :         }
     288             : 
     289         360 :         enumval = lp_parm_enum(SNUM(handle->conn), FRUIT_PARAM_TYPE_NAME,
     290             :                                "resource", fruit_rsrc, FRUIT_RSRC_ADFILE);
     291         360 :         if (enumval == -1) {
     292           0 :                 DEBUG(1, ("value for %s: resource type unknown\n",
     293             :                           FRUIT_PARAM_TYPE_NAME));
     294           0 :                 return -1;
     295             :         }
     296         360 :         config->rsrc = (enum fruit_rsrc)enumval;
     297             : 
     298         360 :         enumval = lp_parm_enum(SNUM(handle->conn), FRUIT_PARAM_TYPE_NAME,
     299             :                                "metadata", fruit_meta, FRUIT_META_NETATALK);
     300         360 :         if (enumval == -1) {
     301           0 :                 DEBUG(1, ("value for %s: metadata type unknown\n",
     302             :                           FRUIT_PARAM_TYPE_NAME));
     303           0 :                 return -1;
     304             :         }
     305         360 :         config->meta = (enum fruit_meta)enumval;
     306             : 
     307         360 :         enumval = lp_parm_enum(SNUM(handle->conn), FRUIT_PARAM_TYPE_NAME,
     308             :                                "locking", fruit_locking, FRUIT_LOCKING_NONE);
     309         360 :         if (enumval == -1) {
     310           0 :                 DEBUG(1, ("value for %s: locking type unknown\n",
     311             :                           FRUIT_PARAM_TYPE_NAME));
     312           0 :                 return -1;
     313             :         }
     314         360 :         config->locking = (enum fruit_locking)enumval;
     315             : 
     316         360 :         enumval = lp_parm_enum(SNUM(handle->conn), FRUIT_PARAM_TYPE_NAME,
     317             :                                "encoding", fruit_encoding, FRUIT_ENC_PRIVATE);
     318         360 :         if (enumval == -1) {
     319           0 :                 DEBUG(1, ("value for %s: encoding type unknown\n",
     320             :                           FRUIT_PARAM_TYPE_NAME));
     321           0 :                 return -1;
     322             :         }
     323         360 :         config->encoding = (enum fruit_encoding)enumval;
     324             : 
     325         360 :         if (config->rsrc == FRUIT_RSRC_ADFILE) {
     326         276 :                 config->veto_appledouble = lp_parm_bool(SNUM(handle->conn),
     327             :                                                         FRUIT_PARAM_TYPE_NAME,
     328             :                                                         "veto_appledouble",
     329             :                                                         true);
     330             :         }
     331             : 
     332         360 :         config->use_aapl = lp_parm_bool(
     333             :                 -1, FRUIT_PARAM_TYPE_NAME, "aapl", true);
     334             : 
     335         360 :         config->time_machine = lp_parm_bool(
     336         360 :                 SNUM(handle->conn), FRUIT_PARAM_TYPE_NAME, "time machine", false);
     337             : 
     338         360 :         config->unix_info_enabled = lp_parm_bool(
     339             :                 -1, FRUIT_PARAM_TYPE_NAME, "nfs_aces", true);
     340             : 
     341         360 :         config->use_copyfile = lp_parm_bool(-1, FRUIT_PARAM_TYPE_NAME,
     342             :                                            "copyfile", false);
     343             : 
     344         360 :         config->posix_rename = lp_parm_bool(
     345         360 :                 SNUM(handle->conn), FRUIT_PARAM_TYPE_NAME, "posix_rename", true);
     346             : 
     347         360 :         config->aapl_zero_file_id =
     348         360 :             lp_parm_bool(SNUM(handle->conn), FRUIT_PARAM_TYPE_NAME,
     349             :                          "zero_file_id", true);
     350             : 
     351         360 :         config->readdir_attr_rsize = lp_parm_bool(
     352         360 :                 SNUM(handle->conn), "readdir_attr", "aapl_rsize", true);
     353             : 
     354         360 :         config->readdir_attr_finder_info = lp_parm_bool(
     355         360 :                 SNUM(handle->conn), "readdir_attr", "aapl_finder_info", true);
     356             : 
     357         360 :         config->readdir_attr_max_access = lp_parm_bool(
     358         360 :                 SNUM(handle->conn), "readdir_attr", "aapl_max_access", true);
     359             : 
     360         360 :         config->model = lp_parm_const_string(
     361             :                 -1, FRUIT_PARAM_TYPE_NAME, "model", "MacSamba");
     362             : 
     363         360 :         tm_size_str = lp_parm_const_string(
     364         360 :                 SNUM(handle->conn), FRUIT_PARAM_TYPE_NAME,
     365             :                 "time machine max size", NULL);
     366         360 :         if (tm_size_str != NULL) {
     367           8 :                 config->time_machine_max_size = conv_str_size(tm_size_str);
     368             :         }
     369             : 
     370         360 :         config->convert_adouble = lp_parm_bool(
     371         360 :                 SNUM(handle->conn), FRUIT_PARAM_TYPE_NAME,
     372             :                 "convert_adouble", true);
     373             : 
     374         360 :         config->wipe_intentionally_left_blank_rfork = lp_parm_bool(
     375         360 :                 SNUM(handle->conn), FRUIT_PARAM_TYPE_NAME,
     376             :                 "wipe_intentionally_left_blank_rfork", false);
     377             : 
     378         360 :         config->delete_empty_adfiles = lp_parm_bool(
     379         360 :                 SNUM(handle->conn), FRUIT_PARAM_TYPE_NAME,
     380             :                 "delete_empty_adfiles", false);
     381             : 
     382         360 :         config->validate_afpinfo = lp_parm_bool(
     383         360 :                 SNUM(handle->conn), FRUIT_PARAM_TYPE_NAME,
     384             :                 "validate_afpinfo", true);
     385             : 
     386         360 :         SMB_VFS_HANDLE_SET_DATA(handle, config,
     387             :                                 NULL, struct fruit_config_data,
     388             :                                 return -1);
     389             : 
     390         360 :         return 0;
     391             : }
     392             : 
     393         482 : static bool add_fruit_stream(TALLOC_CTX *mem_ctx, unsigned int *num_streams,
     394             :                              struct stream_struct **streams,
     395             :                              const char *name, off_t size,
     396             :                              off_t alloc_size)
     397             : {
     398             :         struct stream_struct *tmp;
     399             : 
     400         482 :         tmp = talloc_realloc(mem_ctx, *streams, struct stream_struct,
     401             :                              (*num_streams)+1);
     402         482 :         if (tmp == NULL) {
     403           0 :                 return false;
     404             :         }
     405             : 
     406         482 :         tmp[*num_streams].name = talloc_asprintf(tmp, "%s:$DATA", name);
     407         482 :         if (tmp[*num_streams].name == NULL) {
     408           0 :                 return false;
     409             :         }
     410             : 
     411         482 :         tmp[*num_streams].size = size;
     412         482 :         tmp[*num_streams].alloc_size = alloc_size;
     413             : 
     414         482 :         *streams = tmp;
     415         482 :         *num_streams += 1;
     416         482 :         return true;
     417             : }
     418             : 
     419        1204 : static bool filter_empty_rsrc_stream(unsigned int *num_streams,
     420             :                                      struct stream_struct **streams)
     421             : {
     422        1204 :         struct stream_struct *tmp = *streams;
     423             :         unsigned int i;
     424             : 
     425        1204 :         if (*num_streams == 0) {
     426           0 :                 return true;
     427             :         }
     428             : 
     429        2502 :         for (i = 0; i < *num_streams; i++) {
     430        1390 :                 if (strequal_m(tmp[i].name, AFPRESOURCE_STREAM)) {
     431          92 :                         break;
     432             :                 }
     433             :         }
     434             : 
     435        1204 :         if (i == *num_streams) {
     436        1112 :                 return true;
     437             :         }
     438             : 
     439          92 :         if (tmp[i].size > 0) {
     440          88 :                 return true;
     441             :         }
     442             : 
     443           4 :         TALLOC_FREE(tmp[i].name);
     444           4 :         ARRAY_DEL_ELEMENT(tmp, i, *num_streams);
     445           4 :         *num_streams -= 1;
     446           4 :         return true;
     447             : }
     448             : 
     449        2642 : static bool del_fruit_stream(TALLOC_CTX *mem_ctx, unsigned int *num_streams,
     450             :                              struct stream_struct **streams,
     451             :                              const char *name)
     452             : {
     453        2642 :         struct stream_struct *tmp = *streams;
     454             :         unsigned int i;
     455             : 
     456        2642 :         if (*num_streams == 0) {
     457         984 :                 return true;
     458             :         }
     459             : 
     460        3388 :         for (i = 0; i < *num_streams; i++) {
     461        1756 :                 if (strequal_m(tmp[i].name, name)) {
     462          26 :                         break;
     463             :                 }
     464             :         }
     465             : 
     466        1658 :         if (i == *num_streams) {
     467        1632 :                 return true;
     468             :         }
     469             : 
     470          26 :         TALLOC_FREE(tmp[i].name);
     471          26 :         ARRAY_DEL_ELEMENT(tmp, i, *num_streams);
     472          26 :         *num_streams -= 1;
     473          26 :         return true;
     474             : }
     475             : 
     476         194 : static bool ad_empty_finderinfo(const struct adouble *ad)
     477             : {
     478             :         int cmp;
     479         194 :         char emptybuf[ADEDLEN_FINDERI] = {0};
     480         194 :         char *fi = NULL;
     481             : 
     482         194 :         fi = ad_get_entry(ad, ADEID_FINDERI);
     483         194 :         if (fi == NULL) {
     484           0 :                 DBG_ERR("Missing FinderInfo in struct adouble [%p]\n", ad);
     485           0 :                 return false;
     486             :         }
     487             : 
     488         194 :         cmp = memcmp(emptybuf, fi, ADEDLEN_FINDERI);
     489         194 :         return (cmp == 0);
     490             : }
     491             : 
     492         540 : static bool ai_empty_finderinfo(const AfpInfo *ai)
     493             : {
     494             :         int cmp;
     495         540 :         char emptybuf[ADEDLEN_FINDERI] = {0};
     496             : 
     497         540 :         cmp = memcmp(emptybuf, &ai->afpi_FinderInfo[0], ADEDLEN_FINDERI);
     498         540 :         return (cmp == 0);
     499             : }
     500             : 
     501             : /**
     502             :  * Update btime with btime from Netatalk
     503             :  **/
     504      121940 : static void update_btime(vfs_handle_struct *handle,
     505             :                          struct smb_filename *smb_fname)
     506             : {
     507             :         uint32_t t;
     508      121940 :         struct timespec creation_time = {0};
     509             :         struct adouble *ad;
     510      121940 :         struct fruit_config_data *config = NULL;
     511             : 
     512      121940 :         SMB_VFS_HANDLE_GET_DATA(handle, config, struct fruit_config_data,
     513             :                                 return);
     514             : 
     515      121940 :         switch (config->meta) {
     516       60080 :         case FRUIT_META_STREAM:
     517       60080 :                 return;
     518       61860 :         case FRUIT_META_NETATALK:
     519             :                 /* Handled below */
     520       61860 :                 break;
     521           0 :         default:
     522           0 :                 DBG_ERR("Unexpected meta config [%d]\n", config->meta);
     523           0 :                 return;
     524             :         }
     525             : 
     526       61860 :         ad = ad_get_meta_fsp(talloc_tos(), handle, smb_fname);
     527       61860 :         if (ad == NULL) {
     528       61496 :                 return;
     529             :         }
     530         364 :         if (ad_getdate(ad, AD_DATE_UNIX | AD_DATE_CREATE, &t) != 0) {
     531           0 :                 TALLOC_FREE(ad);
     532           0 :                 return;
     533             :         }
     534         364 :         TALLOC_FREE(ad);
     535             : 
     536         364 :         creation_time.tv_sec = convert_uint32_t_to_time_t(t);
     537         364 :         update_stat_ex_create_time(&smb_fname->st, creation_time);
     538             : 
     539         364 :         return;
     540             : }
     541             : 
     542             : /**
     543             :  * Map an access mask to a Netatalk single byte byte range lock
     544             :  **/
     545        2352 : static off_t access_to_netatalk_brl(enum apple_fork fork_type,
     546             :                                     uint32_t access_mask)
     547             : {
     548             :         off_t offset;
     549             : 
     550        2352 :         switch (access_mask) {
     551        1178 :         case FILE_READ_DATA:
     552        1178 :                 offset = AD_FILELOCK_OPEN_RD;
     553        1178 :                 break;
     554             : 
     555        1174 :         case FILE_WRITE_DATA:
     556             :         case FILE_APPEND_DATA:
     557        1174 :                 offset = AD_FILELOCK_OPEN_WR;
     558        1174 :                 break;
     559             : 
     560           0 :         default:
     561           0 :                 offset = AD_FILELOCK_OPEN_NONE;
     562           0 :                 break;
     563             :         }
     564             : 
     565        2352 :         if (fork_type == APPLE_FORK_RSRC) {
     566           0 :                 if (offset == AD_FILELOCK_OPEN_NONE) {
     567           0 :                         offset = AD_FILELOCK_RSRC_OPEN_NONE;
     568             :                 } else {
     569           0 :                         offset += 2;
     570             :                 }
     571             :         }
     572             : 
     573        2352 :         return offset;
     574             : }
     575             : 
     576             : /**
     577             :  * Map a deny mode to a Netatalk brl
     578             :  **/
     579        1476 : static off_t denymode_to_netatalk_brl(enum apple_fork fork_type,
     580             :                                       uint32_t deny_mode)
     581             : {
     582        1476 :         off_t offset = 0;
     583             : 
     584        1476 :         switch (deny_mode) {
     585         738 :         case DENY_READ:
     586         738 :                 offset = AD_FILELOCK_DENY_RD;
     587         738 :                 break;
     588             : 
     589         738 :         case DENY_WRITE:
     590         738 :                 offset = AD_FILELOCK_DENY_WR;
     591         738 :                 break;
     592             : 
     593           0 :         default:
     594           0 :                 smb_panic("denymode_to_netatalk_brl: bad deny mode\n");
     595             :         }
     596             : 
     597        1476 :         if (fork_type == APPLE_FORK_RSRC) {
     598           0 :                 offset += 2;
     599             :         }
     600             : 
     601        1476 :         return offset;
     602             : }
     603             : 
     604             : /**
     605             :  * Call fcntl() with an exclusive F_GETLK request in order to
     606             :  * determine if there's an existing shared lock
     607             :  *
     608             :  * @return true if the requested lock was found or any error occurred
     609             :  *         false if the lock was not found
     610             :  **/
     611        2944 : static bool test_netatalk_lock(files_struct *fsp, off_t in_offset)
     612             : {
     613             :         bool result;
     614        2944 :         off_t offset = in_offset;
     615        2944 :         off_t len = 1;
     616        2944 :         int type = F_WRLCK;
     617        2944 :         pid_t pid = 0;
     618             : 
     619        2944 :         result = SMB_VFS_GETLOCK(fsp, &offset, &len, &type, &pid);
     620        2944 :         if (result == false) {
     621           0 :                 return true;
     622             :         }
     623             : 
     624        2944 :         if (type != F_UNLCK) {
     625          24 :                 return true;
     626             :         }
     627             : 
     628        2920 :         return false;
     629             : }
     630             : 
     631         736 : static NTSTATUS fruit_check_access(vfs_handle_struct *handle,
     632             :                                    files_struct *fsp,
     633             :                                    uint32_t access_mask,
     634             :                                    uint32_t share_mode)
     635             : {
     636         736 :         NTSTATUS status = NT_STATUS_OK;
     637             :         off_t off;
     638         736 :         bool share_for_read = (share_mode & FILE_SHARE_READ);
     639         736 :         bool share_for_write = (share_mode & FILE_SHARE_WRITE);
     640         736 :         bool netatalk_already_open_for_reading = false;
     641         736 :         bool netatalk_already_open_for_writing = false;
     642         736 :         bool netatalk_already_open_with_deny_read = false;
     643         736 :         bool netatalk_already_open_with_deny_write = false;
     644         736 :         struct GUID req_guid = GUID_random();
     645             : 
     646             :         /* FIXME: hardcoded data fork, add resource fork */
     647         736 :         enum apple_fork fork_type = APPLE_FORK_DATA;
     648             : 
     649         736 :         DBG_DEBUG("fruit_check_access: %s, am: %s/%s, sm: 0x%x\n",
     650             :                   fsp_str_dbg(fsp),
     651             :                   access_mask & FILE_READ_DATA ? "READ" :"-",
     652             :                   access_mask & FILE_WRITE_DATA ? "WRITE" : "-",
     653             :                   share_mode);
     654             : 
     655         736 :         if (fsp_get_io_fd(fsp) == -1) {
     656           0 :                 return NT_STATUS_OK;
     657             :         }
     658             : 
     659             :         /* Read NetATalk opens and deny modes on the file. */
     660         736 :         netatalk_already_open_for_reading = test_netatalk_lock(fsp,
     661             :                                 access_to_netatalk_brl(fork_type,
     662             :                                         FILE_READ_DATA));
     663             : 
     664         736 :         netatalk_already_open_with_deny_read = test_netatalk_lock(fsp,
     665             :                                 denymode_to_netatalk_brl(fork_type,
     666             :                                         DENY_READ));
     667             : 
     668         736 :         netatalk_already_open_for_writing = test_netatalk_lock(fsp,
     669             :                                 access_to_netatalk_brl(fork_type,
     670             :                                         FILE_WRITE_DATA));
     671             : 
     672         736 :         netatalk_already_open_with_deny_write = test_netatalk_lock(fsp,
     673             :                                 denymode_to_netatalk_brl(fork_type,
     674             :                                         DENY_WRITE));
     675             : 
     676             :         /* If there are any conflicts - sharing violation. */
     677         736 :         if ((access_mask & FILE_READ_DATA) &&
     678             :                         netatalk_already_open_with_deny_read) {
     679           0 :                 return NT_STATUS_SHARING_VIOLATION;
     680             :         }
     681             : 
     682         736 :         if (!share_for_read &&
     683             :                         netatalk_already_open_for_reading) {
     684           0 :                 return NT_STATUS_SHARING_VIOLATION;
     685             :         }
     686             : 
     687         736 :         if ((access_mask & FILE_WRITE_DATA) &&
     688             :                         netatalk_already_open_with_deny_write) {
     689           0 :                 return NT_STATUS_SHARING_VIOLATION;
     690             :         }
     691             : 
     692         736 :         if (!share_for_write &&
     693             :                         netatalk_already_open_for_writing) {
     694           2 :                 return NT_STATUS_SHARING_VIOLATION;
     695             :         }
     696             : 
     697         734 :         if (!(access_mask & FILE_READ_DATA)) {
     698             :                 /*
     699             :                  * Nothing we can do here, we need read access
     700             :                  * to set locks.
     701             :                  */
     702         292 :                 return NT_STATUS_OK;
     703             :         }
     704             : 
     705             :         /* Set NetAtalk locks matching our access */
     706         442 :         if (access_mask & FILE_READ_DATA) {
     707         442 :                 off = access_to_netatalk_brl(fork_type, FILE_READ_DATA);
     708         442 :                 req_guid.time_hi_and_version = __LINE__;
     709         442 :                 status = do_lock(
     710             :                         fsp,
     711             :                         talloc_tos(),
     712             :                         &req_guid,
     713         442 :                         fsp->op->global->open_persistent_id,
     714             :                         1,
     715             :                         off,
     716             :                         READ_LOCK,
     717             :                         POSIX_LOCK,
     718             :                         NULL,
     719             :                         NULL);
     720             : 
     721         442 :                 if (!NT_STATUS_IS_OK(status))  {
     722           0 :                         return status;
     723             :                 }
     724             :         }
     725             : 
     726         442 :         if (!share_for_read) {
     727           2 :                 off = denymode_to_netatalk_brl(fork_type, DENY_READ);
     728           2 :                 req_guid.time_hi_and_version = __LINE__;
     729           2 :                 status = do_lock(
     730             :                         fsp,
     731             :                         talloc_tos(),
     732             :                         &req_guid,
     733           2 :                         fsp->op->global->open_persistent_id,
     734             :                         1,
     735             :                         off,
     736             :                         READ_LOCK,
     737             :                         POSIX_LOCK,
     738             :                         NULL,
     739             :                         NULL);
     740             : 
     741           2 :                 if (!NT_STATUS_IS_OK(status)) {
     742           0 :                         return status;
     743             :                 }
     744             :         }
     745             : 
     746         442 :         if (access_mask & FILE_WRITE_DATA) {
     747         438 :                 off = access_to_netatalk_brl(fork_type, FILE_WRITE_DATA);
     748         438 :                 req_guid.time_hi_and_version = __LINE__;
     749         438 :                 status = do_lock(
     750             :                         fsp,
     751             :                         talloc_tos(),
     752             :                         &req_guid,
     753         438 :                         fsp->op->global->open_persistent_id,
     754             :                         1,
     755             :                         off,
     756             :                         READ_LOCK,
     757             :                         POSIX_LOCK,
     758             :                         NULL,
     759             :                         NULL);
     760             : 
     761         438 :                 if (!NT_STATUS_IS_OK(status)) {
     762           0 :                         return status;
     763             :                 }
     764             :         }
     765             : 
     766         442 :         if (!share_for_write) {
     767           2 :                 off = denymode_to_netatalk_brl(fork_type, DENY_WRITE);
     768           2 :                 req_guid.time_hi_and_version = __LINE__;
     769           2 :                 status = do_lock(
     770             :                         fsp,
     771             :                         talloc_tos(),
     772             :                         &req_guid,
     773           2 :                         fsp->op->global->open_persistent_id,
     774             :                         1,
     775             :                         off,
     776             :                         READ_LOCK,
     777             :                         POSIX_LOCK,
     778             :                         NULL,
     779             :                         NULL);
     780             : 
     781           2 :                 if (!NT_STATUS_IS_OK(status)) {
     782           0 :                         return status;
     783             :                 }
     784             :         }
     785             : 
     786         442 :         return NT_STATUS_OK;
     787             : }
     788             : 
     789       10598 : static NTSTATUS check_aapl(vfs_handle_struct *handle,
     790             :                            struct smb_request *req,
     791             :                            const struct smb2_create_blobs *in_context_blobs,
     792             :                            struct smb2_create_blobs *out_context_blobs)
     793             : {
     794             :         struct fruit_config_data *config;
     795             :         NTSTATUS status;
     796       10598 :         struct smb2_create_blob *aapl = NULL;
     797             :         uint32_t cmd;
     798             :         bool ok;
     799             :         uint8_t p[16];
     800       10598 :         DATA_BLOB blob = data_blob_talloc(req, NULL, 0);
     801             :         uint64_t req_bitmap, client_caps;
     802       10598 :         uint64_t server_caps = SMB2_CRTCTX_AAPL_UNIX_BASED;
     803             :         smb_ucs2_t *model;
     804             :         size_t modellen;
     805             : 
     806       10598 :         SMB_VFS_HANDLE_GET_DATA(handle, config, struct fruit_config_data,
     807             :                                 return NT_STATUS_UNSUCCESSFUL);
     808             : 
     809       10598 :         if (!config->use_aapl
     810       10598 :             || in_context_blobs == NULL
     811        9972 :             || out_context_blobs == NULL) {
     812         626 :                 return NT_STATUS_OK;
     813             :         }
     814             : 
     815        9972 :         aapl = smb2_create_blob_find(in_context_blobs,
     816             :                                      SMB2_CREATE_TAG_AAPL);
     817        9972 :         if (aapl == NULL) {
     818        9844 :                 return NT_STATUS_OK;
     819             :         }
     820             : 
     821         128 :         if (aapl->data.length != 24) {
     822           0 :                 DEBUG(1, ("unexpected AAPL ctxt length: %ju\n",
     823             :                           (uintmax_t)aapl->data.length));
     824           0 :                 return NT_STATUS_INVALID_PARAMETER;
     825             :         }
     826             : 
     827         128 :         cmd = IVAL(aapl->data.data, 0);
     828         128 :         if (cmd != SMB2_CRTCTX_AAPL_SERVER_QUERY) {
     829           0 :                 DEBUG(1, ("unsupported AAPL cmd: %d\n", cmd));
     830           0 :                 return NT_STATUS_INVALID_PARAMETER;
     831             :         }
     832             : 
     833         128 :         req_bitmap = BVAL(aapl->data.data, 8);
     834         128 :         client_caps = BVAL(aapl->data.data, 16);
     835             : 
     836         128 :         SIVAL(p, 0, SMB2_CRTCTX_AAPL_SERVER_QUERY);
     837         128 :         SIVAL(p, 4, 0);
     838         128 :         SBVAL(p, 8, req_bitmap);
     839         128 :         ok = data_blob_append(req, &blob, p, 16);
     840         128 :         if (!ok) {
     841           0 :                 return NT_STATUS_UNSUCCESSFUL;
     842             :         }
     843             : 
     844         128 :         if (req_bitmap & SMB2_CRTCTX_AAPL_SERVER_CAPS) {
     845         128 :                 if ((client_caps & SMB2_CRTCTX_AAPL_SUPPORTS_READ_DIR_ATTR) &&
     846         120 :                     (handle->conn->fs_capabilities & FILE_NAMED_STREAMS)) {
     847         120 :                         server_caps |= SMB2_CRTCTX_AAPL_SUPPORTS_READ_DIR_ATTR;
     848         120 :                         config->readdir_attr_enabled = true;
     849             :                 }
     850             : 
     851         128 :                 if (config->use_copyfile) {
     852         128 :                         server_caps |= SMB2_CRTCTX_AAPL_SUPPORTS_OSX_COPYFILE;
     853         128 :                         config->copyfile_enabled = true;
     854             :                 }
     855             : 
     856             :                 /*
     857             :                  * The client doesn't set the flag, so we can't check
     858             :                  * for it and just set it unconditionally
     859             :                  */
     860         128 :                 if (config->unix_info_enabled) {
     861         128 :                         server_caps |= SMB2_CRTCTX_AAPL_SUPPORTS_NFS_ACE;
     862             :                 }
     863             : 
     864         128 :                 SBVAL(p, 0, server_caps);
     865         128 :                 ok = data_blob_append(req, &blob, p, 8);
     866         128 :                 if (!ok) {
     867           0 :                         return NT_STATUS_UNSUCCESSFUL;
     868             :                 }
     869             :         }
     870             : 
     871         128 :         if (req_bitmap & SMB2_CRTCTX_AAPL_VOLUME_CAPS) {
     872         120 :                 int val = lp_case_sensitive(SNUM(handle->conn));
     873         120 :                 uint64_t caps = 0;
     874             : 
     875         120 :                 switch (val) {
     876         120 :                 case Auto:
     877         120 :                         break;
     878             : 
     879           0 :                 case True:
     880           0 :                         caps |= SMB2_CRTCTX_AAPL_CASE_SENSITIVE;
     881           0 :                         break;
     882             : 
     883           0 :                 default:
     884           0 :                         break;
     885             :                 }
     886             : 
     887         120 :                 if (config->time_machine) {
     888           2 :                         caps |= SMB2_CRTCTX_AAPL_FULL_SYNC;
     889             :                 }
     890             : 
     891         120 :                 SBVAL(p, 0, caps);
     892             : 
     893         120 :                 ok = data_blob_append(req, &blob, p, 8);
     894         120 :                 if (!ok) {
     895           0 :                         return NT_STATUS_UNSUCCESSFUL;
     896             :                 }
     897             :         }
     898             : 
     899         128 :         if (req_bitmap & SMB2_CRTCTX_AAPL_MODEL_INFO) {
     900         120 :                 ok = convert_string_talloc(req,
     901             :                                            CH_UNIX, CH_UTF16LE,
     902         120 :                                            config->model, strlen(config->model),
     903             :                                            &model, &modellen);
     904         120 :                 if (!ok) {
     905           0 :                         return NT_STATUS_UNSUCCESSFUL;
     906             :                 }
     907             : 
     908         120 :                 SIVAL(p, 0, 0);
     909         120 :                 SIVAL(p + 4, 0, modellen);
     910         120 :                 ok = data_blob_append(req, &blob, p, 8);
     911         120 :                 if (!ok) {
     912           0 :                         talloc_free(model);
     913           0 :                         return NT_STATUS_UNSUCCESSFUL;
     914             :                 }
     915             : 
     916         120 :                 ok = data_blob_append(req, &blob, model, modellen);
     917         120 :                 talloc_free(model);
     918         120 :                 if (!ok) {
     919           0 :                         return NT_STATUS_UNSUCCESSFUL;
     920             :                 }
     921             :         }
     922             : 
     923         128 :         status = smb2_create_blob_add(out_context_blobs,
     924             :                                       out_context_blobs,
     925             :                                       SMB2_CREATE_TAG_AAPL,
     926             :                                       blob);
     927         128 :         if (NT_STATUS_IS_OK(status)) {
     928         128 :                 global_fruit_config.nego_aapl = true;
     929             :         }
     930             : 
     931         128 :         return status;
     932             : }
     933             : 
     934         238 : static bool readdir_attr_meta_finderi_stream(
     935             :         struct vfs_handle_struct *handle,
     936             :         const struct smb_filename *smb_fname,
     937             :         AfpInfo *ai)
     938             : {
     939         238 :         struct smb_filename *stream_name = NULL;
     940         238 :         files_struct *fsp = NULL;
     941             :         ssize_t nread;
     942             :         NTSTATUS status;
     943             :         bool ok;
     944             :         uint8_t buf[AFP_INFO_SIZE];
     945             : 
     946         238 :         status = synthetic_pathref(talloc_tos(),
     947         238 :                                    handle->conn->cwd_fsp,
     948         238 :                                    smb_fname->base_name,
     949             :                                    AFPINFO_STREAM_NAME,
     950             :                                    NULL,
     951         238 :                                    smb_fname->twrp,
     952         238 :                                    smb_fname->flags,
     953             :                                    &stream_name);
     954         238 :         if (!NT_STATUS_IS_OK(status)) {
     955         206 :                 return false;
     956             :         }
     957             : 
     958          32 :         status = SMB_VFS_CREATE_FILE(
     959             :                 handle->conn,                           /* conn */
     960             :                 NULL,                                   /* req */
     961             :                 NULL,                                   /* dirfsp */
     962             :                 stream_name,                            /* fname */
     963             :                 FILE_READ_DATA,                         /* access_mask */
     964             :                 (FILE_SHARE_READ | FILE_SHARE_WRITE |   /* share_access */
     965             :                         FILE_SHARE_DELETE),
     966             :                 FILE_OPEN,                              /* create_disposition*/
     967             :                 0,                                      /* create_options */
     968             :                 0,                                      /* file_attributes */
     969             :                 INTERNAL_OPEN_ONLY,                     /* oplock_request */
     970             :                 NULL,                                   /* lease */
     971             :                 0,                                      /* allocation_size */
     972             :                 0,                                      /* private_flags */
     973             :                 NULL,                                   /* sd */
     974             :                 NULL,                                   /* ea_list */
     975             :                 &fsp,                                   /* result */
     976             :                 NULL,                                   /* pinfo */
     977             :                 NULL, NULL);                            /* create context */
     978             : 
     979          32 :         TALLOC_FREE(stream_name);
     980             : 
     981          32 :         if (!NT_STATUS_IS_OK(status)) {
     982           0 :                 return false;
     983             :         }
     984             : 
     985          32 :         nread = SMB_VFS_PREAD(fsp, &buf[0], AFP_INFO_SIZE, 0);
     986          32 :         if (nread != AFP_INFO_SIZE) {
     987           0 :                 DBG_ERR("short read [%s] [%zd/%d]\n",
     988             :                         smb_fname_str_dbg(stream_name), nread, AFP_INFO_SIZE);
     989           0 :                 ok = false;
     990           0 :                 goto fail;
     991             :         }
     992             : 
     993          32 :         memcpy(&ai->afpi_FinderInfo[0], &buf[AFP_OFF_FinderInfo],
     994             :                AFP_FinderSize);
     995             : 
     996          32 :         ok = true;
     997             : 
     998          32 : fail:
     999          32 :         if (fsp != NULL) {
    1000          32 :                 close_file_free(NULL, &fsp, NORMAL_CLOSE);
    1001             :         }
    1002             : 
    1003          32 :         return ok;
    1004             : }
    1005             : 
    1006          66 : static bool readdir_attr_meta_finderi_netatalk(
    1007             :         struct vfs_handle_struct *handle,
    1008             :         const struct smb_filename *smb_fname,
    1009             :         AfpInfo *ai)
    1010             : {
    1011          66 :         struct adouble *ad = NULL;
    1012          66 :         char *p = NULL;
    1013             : 
    1014          66 :         ad = ad_get_meta_fsp(talloc_tos(), handle, smb_fname);
    1015          66 :         if (ad == NULL) {
    1016          58 :                 return false;
    1017             :         }
    1018             : 
    1019           8 :         p = ad_get_entry(ad, ADEID_FINDERI);
    1020           8 :         if (p == NULL) {
    1021           0 :                 DBG_ERR("No ADEID_FINDERI for [%s]\n", smb_fname->base_name);
    1022           0 :                 TALLOC_FREE(ad);
    1023           0 :                 return false;
    1024             :         }
    1025             : 
    1026           8 :         memcpy(&ai->afpi_FinderInfo[0], p, AFP_FinderSize);
    1027           8 :         TALLOC_FREE(ad);
    1028           8 :         return true;
    1029             : }
    1030             : 
    1031         304 : static bool readdir_attr_meta_finderi(struct vfs_handle_struct *handle,
    1032             :                                       const struct smb_filename *smb_fname,
    1033             :                                       struct readdir_attr_data *attr_data)
    1034             : {
    1035         304 :         struct fruit_config_data *config = NULL;
    1036             :         uint32_t date_added;
    1037         304 :         AfpInfo ai = {0};
    1038             :         bool ok;
    1039             : 
    1040         304 :         SMB_VFS_HANDLE_GET_DATA(handle, config,
    1041             :                                 struct fruit_config_data,
    1042             :                                 return false);
    1043             : 
    1044         304 :         switch (config->meta) {
    1045          66 :         case FRUIT_META_NETATALK:
    1046          66 :                 ok = readdir_attr_meta_finderi_netatalk(
    1047             :                         handle, smb_fname, &ai);
    1048          66 :                 break;
    1049             : 
    1050         238 :         case FRUIT_META_STREAM:
    1051         238 :                 ok = readdir_attr_meta_finderi_stream(
    1052             :                         handle, smb_fname, &ai);
    1053         238 :                 break;
    1054             : 
    1055           0 :         default:
    1056           0 :                 DBG_ERR("Unexpected meta config [%d]\n", config->meta);
    1057           0 :                 return false;
    1058             :         }
    1059             : 
    1060         304 :         if (!ok) {
    1061             :                 /* Don't bother with errors, it's likely ENOENT */
    1062         264 :                 return true;
    1063             :         }
    1064             : 
    1065          40 :         if (S_ISREG(smb_fname->st.st_ex_mode)) {
    1066             :                 /* finder_type */
    1067          40 :                 memcpy(&attr_data->attr_data.aapl.finder_info[0],
    1068             :                        &ai.afpi_FinderInfo[0], 4);
    1069             : 
    1070             :                 /* finder_creator */
    1071          40 :                 memcpy(&attr_data->attr_data.aapl.finder_info[0] + 4,
    1072             :                        &ai.afpi_FinderInfo[4], 4);
    1073             :         }
    1074             : 
    1075             :         /* finder_flags */
    1076          40 :         memcpy(&attr_data->attr_data.aapl.finder_info[0] + 8,
    1077             :                &ai.afpi_FinderInfo[8], 2);
    1078             : 
    1079             :         /* finder_ext_flags */
    1080          40 :         memcpy(&attr_data->attr_data.aapl.finder_info[0] + 10,
    1081             :                &ai.afpi_FinderInfo[24], 2);
    1082             : 
    1083             :         /* creation date */
    1084          40 :         date_added = convert_time_t_to_uint32_t(
    1085          40 :                 smb_fname->st.st_ex_btime.tv_sec - AD_DATE_DELTA);
    1086             : 
    1087          40 :         RSIVAL(&attr_data->attr_data.aapl.finder_info[0], 12, date_added);
    1088             : 
    1089          40 :         return true;
    1090             : }
    1091             : 
    1092         240 : static uint64_t readdir_attr_rfork_size_adouble(
    1093             :         struct vfs_handle_struct *handle,
    1094             :         const struct smb_filename *smb_fname)
    1095             : {
    1096         240 :         struct adouble *ad = NULL;
    1097             :         uint64_t rfork_size;
    1098             : 
    1099         240 :         ad = ad_get(talloc_tos(), handle, smb_fname,
    1100             :                     ADOUBLE_RSRC);
    1101         240 :         if (ad == NULL) {
    1102         208 :                 return 0;
    1103             :         }
    1104             : 
    1105          32 :         rfork_size = ad_getentrylen(ad, ADEID_RFORK);
    1106          32 :         TALLOC_FREE(ad);
    1107             : 
    1108          32 :         return rfork_size;
    1109             : }
    1110             : 
    1111          64 : static uint64_t readdir_attr_rfork_size_stream(
    1112             :         struct vfs_handle_struct *handle,
    1113             :         const struct smb_filename *smb_fname)
    1114             : {
    1115          64 :         struct smb_filename *stream_name = NULL;
    1116             :         int ret;
    1117             :         uint64_t rfork_size;
    1118             : 
    1119          64 :         stream_name = synthetic_smb_fname(talloc_tos(),
    1120          64 :                                           smb_fname->base_name,
    1121             :                                           AFPRESOURCE_STREAM_NAME,
    1122             :                                           NULL,
    1123          64 :                                           smb_fname->twrp,
    1124             :                                           0);
    1125          64 :         if (stream_name == NULL) {
    1126           0 :                 return 0;
    1127             :         }
    1128             : 
    1129          64 :         ret = SMB_VFS_STAT(handle->conn, stream_name);
    1130          64 :         if (ret != 0) {
    1131          56 :                 TALLOC_FREE(stream_name);
    1132          56 :                 return 0;
    1133             :         }
    1134             : 
    1135           8 :         rfork_size = stream_name->st.st_ex_size;
    1136           8 :         TALLOC_FREE(stream_name);
    1137             : 
    1138           8 :         return rfork_size;
    1139             : }
    1140             : 
    1141         304 : static uint64_t readdir_attr_rfork_size(struct vfs_handle_struct *handle,
    1142             :                                         const struct smb_filename *smb_fname)
    1143             : {
    1144         304 :         struct fruit_config_data *config = NULL;
    1145             :         uint64_t rfork_size;
    1146             : 
    1147         304 :         SMB_VFS_HANDLE_GET_DATA(handle, config,
    1148             :                                 struct fruit_config_data,
    1149             :                                 return 0);
    1150             : 
    1151         304 :         switch (config->rsrc) {
    1152         240 :         case FRUIT_RSRC_ADFILE:
    1153         240 :                 rfork_size = readdir_attr_rfork_size_adouble(handle,
    1154             :                                                              smb_fname);
    1155         240 :                 break;
    1156             : 
    1157          64 :         case FRUIT_RSRC_XATTR:
    1158             :         case FRUIT_RSRC_STREAM:
    1159          64 :                 rfork_size = readdir_attr_rfork_size_stream(handle,
    1160             :                                                             smb_fname);
    1161          64 :                 break;
    1162             : 
    1163           0 :         default:
    1164           0 :                 DBG_ERR("Unexpected rsrc config [%d]\n", config->rsrc);
    1165           0 :                 rfork_size = 0;
    1166           0 :                 break;
    1167             :         }
    1168             : 
    1169         304 :         return rfork_size;
    1170             : }
    1171             : 
    1172         304 : static NTSTATUS readdir_attr_macmeta(struct vfs_handle_struct *handle,
    1173             :                                      const struct smb_filename *smb_fname,
    1174             :                                      struct readdir_attr_data *attr_data)
    1175             : {
    1176         304 :         NTSTATUS status = NT_STATUS_OK;
    1177         304 :         struct fruit_config_data *config = NULL;
    1178             :         bool ok;
    1179             : 
    1180         304 :         SMB_VFS_HANDLE_GET_DATA(handle, config,
    1181             :                                 struct fruit_config_data,
    1182             :                                 return NT_STATUS_UNSUCCESSFUL);
    1183             : 
    1184             : 
    1185             :         /* Ensure we return a default value in the creation_date field */
    1186         304 :         RSIVAL(&attr_data->attr_data.aapl.finder_info, 12, AD_DATE_START);
    1187             : 
    1188             :         /*
    1189             :          * Resource fork length
    1190             :          */
    1191             : 
    1192         304 :         if (config->readdir_attr_rsize) {
    1193             :                 uint64_t rfork_size;
    1194             : 
    1195         304 :                 rfork_size = readdir_attr_rfork_size(handle, smb_fname);
    1196         304 :                 attr_data->attr_data.aapl.rfork_size = rfork_size;
    1197             :         }
    1198             : 
    1199             :         /*
    1200             :          * FinderInfo
    1201             :          */
    1202             : 
    1203         304 :         if (config->readdir_attr_finder_info) {
    1204         304 :                 ok = readdir_attr_meta_finderi(handle, smb_fname, attr_data);
    1205         304 :                 if (!ok) {
    1206           0 :                         status = NT_STATUS_INTERNAL_ERROR;
    1207             :                 }
    1208             :         }
    1209             : 
    1210         304 :         return status;
    1211             : }
    1212             : 
    1213        5076 : static NTSTATUS remove_virtual_nfs_aces(struct security_descriptor *psd)
    1214             : {
    1215             :         NTSTATUS status;
    1216             :         uint32_t i;
    1217             : 
    1218        5076 :         if (psd->dacl == NULL) {
    1219           0 :                 return NT_STATUS_OK;
    1220             :         }
    1221             : 
    1222       26968 :         for (i = 0; i < psd->dacl->num_aces; i++) {
    1223             :                 /* MS NFS style mode/uid/gid */
    1224       21892 :                 int cmp = dom_sid_compare_domain(
    1225             :                                 &global_sid_Unix_NFS,
    1226       21892 :                                 &psd->dacl->aces[i].trustee);
    1227       21892 :                 if (cmp != 0) {
    1228             :                         /* Normal ACE entry. */
    1229       21884 :                         continue;
    1230             :                 }
    1231             : 
    1232             :                 /*
    1233             :                  * security_descriptor_dacl_del()
    1234             :                  * *must* return NT_STATUS_OK as we know
    1235             :                  * we have something to remove.
    1236             :                  */
    1237             : 
    1238           8 :                 status = security_descriptor_dacl_del(psd,
    1239           8 :                                 &psd->dacl->aces[i].trustee);
    1240           8 :                 if (!NT_STATUS_IS_OK(status)) {
    1241           0 :                         DBG_WARNING("failed to remove MS NFS style ACE: %s\n",
    1242             :                                 nt_errstr(status));
    1243           0 :                         return status;
    1244             :                 }
    1245             : 
    1246             :                 /*
    1247             :                  * security_descriptor_dacl_del() may delete more
    1248             :                  * then one entry subsequent to this one if the
    1249             :                  * SID matches, but we only need to ensure that
    1250             :                  * we stay looking at the same element in the array.
    1251             :                  */
    1252           8 :                 i--;
    1253             :         }
    1254        5076 :         return NT_STATUS_OK;
    1255             : }
    1256             : 
    1257             : /* Search MS NFS style ACE with UNIX mode */
    1258        1158 : static NTSTATUS check_ms_nfs(vfs_handle_struct *handle,
    1259             :                              files_struct *fsp,
    1260             :                              struct security_descriptor *psd,
    1261             :                              mode_t *pmode,
    1262             :                              bool *pdo_chmod)
    1263             : {
    1264             :         uint32_t i;
    1265        1158 :         struct fruit_config_data *config = NULL;
    1266             : 
    1267        1158 :         *pdo_chmod = false;
    1268             : 
    1269        1158 :         SMB_VFS_HANDLE_GET_DATA(handle, config,
    1270             :                                 struct fruit_config_data,
    1271             :                                 return NT_STATUS_UNSUCCESSFUL);
    1272             : 
    1273        1158 :         if (!global_fruit_config.nego_aapl) {
    1274         796 :                 return NT_STATUS_OK;
    1275             :         }
    1276         362 :         if (psd->dacl == NULL || !config->unix_info_enabled) {
    1277           0 :                 return NT_STATUS_OK;
    1278             :         }
    1279             : 
    1280        1552 :         for (i = 0; i < psd->dacl->num_aces; i++) {
    1281        1198 :                 if (dom_sid_compare_domain(
    1282             :                             &global_sid_Unix_NFS_Mode,
    1283        1198 :                             &psd->dacl->aces[i].trustee) == 0) {
    1284           8 :                         *pmode = (mode_t)psd->dacl->aces[i].trustee.sub_auths[2];
    1285           8 :                         *pmode &= (S_IRWXU | S_IRWXG | S_IRWXO);
    1286           8 :                         *pdo_chmod = true;
    1287             : 
    1288           8 :                         DEBUG(10, ("MS NFS chmod request %s, %04o\n",
    1289             :                                    fsp_str_dbg(fsp), (unsigned)(*pmode)));
    1290           8 :                         break;
    1291             :                 }
    1292             :         }
    1293             : 
    1294             :         /*
    1295             :          * Remove any incoming virtual ACE entries generated by
    1296             :          * fruit_fget_nt_acl().
    1297             :          */
    1298             : 
    1299         362 :         return remove_virtual_nfs_aces(psd);
    1300             : }
    1301             : 
    1302             : /****************************************************************************
    1303             :  * VFS ops
    1304             :  ****************************************************************************/
    1305             : 
    1306         360 : static int fruit_connect(vfs_handle_struct *handle,
    1307             :                          const char *service,
    1308             :                          const char *user)
    1309             : {
    1310             :         int rc;
    1311         360 :         char *list = NULL, *newlist = NULL;
    1312             :         struct fruit_config_data *config;
    1313             :         const struct loadparm_substitution *lp_sub =
    1314         360 :                 loadparm_s3_global_substitution();
    1315             : 
    1316         360 :         DEBUG(10, ("fruit_connect\n"));
    1317             : 
    1318         360 :         rc = SMB_VFS_NEXT_CONNECT(handle, service, user);
    1319         360 :         if (rc < 0) {
    1320           0 :                 return rc;
    1321             :         }
    1322             : 
    1323         360 :         rc = init_fruit_config(handle);
    1324         360 :         if (rc != 0) {
    1325           0 :                 return rc;
    1326             :         }
    1327             : 
    1328         360 :         SMB_VFS_HANDLE_GET_DATA(handle, config,
    1329             :                                 struct fruit_config_data, return -1);
    1330             : 
    1331         360 :         if (config->veto_appledouble) {
    1332          18 :                 list = lp_veto_files(talloc_tos(), lp_sub, SNUM(handle->conn));
    1333             : 
    1334          18 :                 if (list) {
    1335          18 :                         if (strstr(list, "/" ADOUBLE_NAME_PREFIX "*/") == NULL) {
    1336          18 :                                 newlist = talloc_asprintf(
    1337             :                                         list,
    1338             :                                         "%s/" ADOUBLE_NAME_PREFIX "*/",
    1339             :                                         list);
    1340          18 :                                 lp_do_parameter(SNUM(handle->conn),
    1341             :                                                 "veto files",
    1342             :                                                 newlist);
    1343             :                         }
    1344             :                 } else {
    1345           0 :                         lp_do_parameter(SNUM(handle->conn),
    1346             :                                         "veto files",
    1347             :                                         "/" ADOUBLE_NAME_PREFIX "*/");
    1348             :                 }
    1349             : 
    1350          18 :                 TALLOC_FREE(list);
    1351             :         }
    1352             : 
    1353         360 :         if (config->encoding == FRUIT_ENC_NATIVE) {
    1354         100 :                 lp_do_parameter(SNUM(handle->conn),
    1355             :                                 "catia:mappings",
    1356             :                                 macos_string_replace_map);
    1357             :         }
    1358             : 
    1359         360 :         if (config->time_machine) {
    1360           8 :                 DBG_NOTICE("Enabling durable handles for Time Machine "
    1361             :                            "support on [%s]\n", service);
    1362           8 :                 lp_do_parameter(SNUM(handle->conn), "durable handles", "yes");
    1363           8 :                 lp_do_parameter(SNUM(handle->conn), "kernel oplocks", "no");
    1364           8 :                 lp_do_parameter(SNUM(handle->conn), "kernel share modes", "no");
    1365           8 :                 if (!lp_strict_sync(SNUM(handle->conn))) {
    1366           0 :                         DBG_WARNING("Time Machine without strict sync is not "
    1367             :                                     "recommended!\n");
    1368             :                 }
    1369           8 :                 lp_do_parameter(SNUM(handle->conn), "posix locking", "no");
    1370             :         }
    1371             : 
    1372         360 :         return rc;
    1373             : }
    1374             : 
    1375         822 : static void fio_ref_destroy_fn(void *p_data)
    1376             : {
    1377         822 :         struct fio *ref_fio = (struct fio *)p_data;
    1378         822 :         if (ref_fio->real_fio != NULL) {
    1379         822 :                 SMB_ASSERT(ref_fio->real_fio->ad_fsp == ref_fio->fsp);
    1380         822 :                 ref_fio->real_fio->ad_fsp = NULL;
    1381         822 :                 ref_fio->real_fio = NULL;
    1382             :         }
    1383         822 : }
    1384             : 
    1385        6128 : static void fio_close_ad_fsp(struct fio *fio)
    1386             : {
    1387        6128 :         if (fio->ad_fsp != NULL) {
    1388         822 :                 fd_close(fio->ad_fsp);
    1389         822 :                 file_free(NULL, fio->ad_fsp);
    1390             :                 /* fio_ref_destroy_fn() should have cleared this */
    1391         822 :                 SMB_ASSERT(fio->ad_fsp == NULL);
    1392             :         }
    1393        6128 : }
    1394             : 
    1395        5318 : static void fio_destroy_fn(void *p_data)
    1396             : {
    1397        5318 :         struct fio *fio = (struct fio *)p_data;
    1398        5318 :         fio_close_ad_fsp(fio);
    1399        5318 : }
    1400             : 
    1401        4012 : static int fruit_open_meta_stream(vfs_handle_struct *handle,
    1402             :                                   const struct files_struct *dirfsp,
    1403             :                                   const struct smb_filename *smb_fname,
    1404             :                                   files_struct *fsp,
    1405             :                                   int flags,
    1406             :                                   mode_t mode)
    1407             : {
    1408        4012 :         struct fruit_config_data *config = NULL;
    1409        4012 :         struct fio *fio = NULL;
    1410        4012 :         struct vfs_open_how how = {
    1411        4012 :                 .flags = flags & ~O_CREAT,
    1412             :                 .mode = mode,
    1413             :         };
    1414             :         int fd;
    1415             : 
    1416        4012 :         DBG_DEBUG("Path [%s]\n", smb_fname_str_dbg(smb_fname));
    1417             : 
    1418        4012 :         SMB_VFS_HANDLE_GET_DATA(handle, config,
    1419             :                                 struct fruit_config_data, return -1);
    1420             : 
    1421        4012 :         fio = VFS_ADD_FSP_EXTENSION(handle, fsp, struct fio, fio_destroy_fn);
    1422        4012 :         fio->handle = handle;
    1423        4012 :         fio->fsp = fsp;
    1424        4012 :         fio->type = ADOUBLE_META;
    1425        4012 :         fio->config = config;
    1426             : 
    1427        4012 :         fd = SMB_VFS_NEXT_OPENAT(handle,
    1428             :                                  dirfsp,
    1429             :                                  smb_fname,
    1430             :                                  fsp,
    1431             :                                  &how);
    1432        4012 :         if (fd != -1) {
    1433        1446 :                 return fd;
    1434             :         }
    1435             : 
    1436        2566 :         if (!(flags & O_CREAT)) {
    1437        1464 :                 VFS_REMOVE_FSP_EXTENSION(handle, fsp);
    1438        1464 :                 return -1;
    1439             :         }
    1440             : 
    1441        1102 :         fd = vfs_fake_fd();
    1442        1102 :         if (fd == -1) {
    1443           0 :                 VFS_REMOVE_FSP_EXTENSION(handle, fsp);
    1444           0 :                 return -1;
    1445             :         }
    1446             : 
    1447        1102 :         fio->fake_fd = true;
    1448        1102 :         fio->flags = flags;
    1449        1102 :         fio->mode = mode;
    1450             : 
    1451        1102 :         return fd;
    1452             : }
    1453             : 
    1454        1292 : static int fruit_open_meta_netatalk(vfs_handle_struct *handle,
    1455             :                                     const struct files_struct *dirfsp,
    1456             :                                     const struct smb_filename *smb_fname,
    1457             :                                     files_struct *fsp,
    1458             :                                     int flags,
    1459             :                                     mode_t mode)
    1460             : {
    1461        1292 :         struct fruit_config_data *config = NULL;
    1462        1292 :         struct fio *fio = NULL;
    1463        1292 :         struct adouble *ad = NULL;
    1464        1292 :         bool meta_exists = false;
    1465             :         int fd;
    1466             : 
    1467        1292 :         DBG_DEBUG("Path [%s]\n", smb_fname_str_dbg(smb_fname));
    1468             : 
    1469             :         /*
    1470             :          * We know this is a stream open, so fsp->base_fsp must
    1471             :          * already be open.
    1472             :          */
    1473        1292 :         SMB_ASSERT(fsp_is_alternate_stream(fsp));
    1474        1292 :         SMB_ASSERT(fsp->base_fsp->fsp_name->fsp == fsp->base_fsp);
    1475             : 
    1476        1292 :         ad = ad_get(talloc_tos(), handle, fsp->base_fsp->fsp_name, ADOUBLE_META);
    1477        1292 :         if (ad != NULL) {
    1478         504 :                 meta_exists = true;
    1479             :         }
    1480             : 
    1481        1292 :         TALLOC_FREE(ad);
    1482             : 
    1483        1292 :         if (!meta_exists && !(flags & O_CREAT)) {
    1484         420 :                 errno = ENOENT;
    1485         420 :                 return -1;
    1486             :         }
    1487             : 
    1488         872 :         fd = vfs_fake_fd();
    1489         872 :         if (fd == -1) {
    1490           0 :                 return -1;
    1491             :         }
    1492             : 
    1493         872 :         SMB_VFS_HANDLE_GET_DATA(handle, config,
    1494             :                                 struct fruit_config_data, return -1);
    1495             : 
    1496         872 :         fio = VFS_ADD_FSP_EXTENSION(handle, fsp, struct fio, fio_destroy_fn);
    1497         872 :         fio->handle = handle;
    1498         872 :         fio->fsp = fsp;
    1499         872 :         fio->type = ADOUBLE_META;
    1500         872 :         fio->config = config;
    1501         872 :         fio->fake_fd = true;
    1502         872 :         fio->flags = flags;
    1503         872 :         fio->mode = mode;
    1504             : 
    1505         872 :         return fd;
    1506             : }
    1507             : 
    1508        5304 : static int fruit_open_meta(vfs_handle_struct *handle,
    1509             :                            const struct files_struct *dirfsp,
    1510             :                            const struct smb_filename *smb_fname,
    1511             :                            files_struct *fsp, int flags, mode_t mode)
    1512             : {
    1513             :         int fd;
    1514        5304 :         struct fruit_config_data *config = NULL;
    1515             : 
    1516        5304 :         DBG_DEBUG("path [%s]\n", smb_fname_str_dbg(smb_fname));
    1517             : 
    1518        5304 :         SMB_VFS_HANDLE_GET_DATA(handle, config,
    1519             :                                 struct fruit_config_data, return -1);
    1520             : 
    1521        5304 :         switch (config->meta) {
    1522        4012 :         case FRUIT_META_STREAM:
    1523        4012 :                 fd = fruit_open_meta_stream(handle, dirfsp, smb_fname,
    1524             :                                             fsp, flags, mode);
    1525        4012 :                 break;
    1526             : 
    1527        1292 :         case FRUIT_META_NETATALK:
    1528        1292 :                 fd = fruit_open_meta_netatalk(handle, dirfsp, smb_fname,
    1529             :                                               fsp, flags, mode);
    1530        1292 :                 break;
    1531             : 
    1532           0 :         default:
    1533           0 :                 DBG_ERR("Unexpected meta config [%d]\n", config->meta);
    1534           0 :                 return -1;
    1535             :         }
    1536             : 
    1537        5304 :         DBG_DEBUG("path [%s] fd [%d]\n", smb_fname_str_dbg(smb_fname), fd);
    1538             : 
    1539        5304 :         return fd;
    1540             : }
    1541             : 
    1542        1012 : static int fruit_open_rsrc_adouble(vfs_handle_struct *handle,
    1543             :                                    const struct files_struct *dirfsp,
    1544             :                                    const struct smb_filename *smb_fname,
    1545             :                                    files_struct *fsp,
    1546             :                                    int flags,
    1547             :                                    mode_t mode)
    1548             : {
    1549        1012 :         int rc = 0;
    1550        1012 :         struct fruit_config_data *config = NULL;
    1551        1012 :         struct files_struct *ad_fsp = NULL;
    1552        1012 :         struct fio *fio = NULL;
    1553        1012 :         struct fio *ref_fio = NULL;
    1554             :         NTSTATUS status;
    1555        1012 :         int fd = -1;
    1556             : 
    1557        1012 :         SMB_VFS_HANDLE_GET_DATA(handle, config,
    1558             :                                 struct fruit_config_data, return -1);
    1559             : 
    1560        1012 :         if ((!(flags & O_CREAT)) &&
    1561         874 :             S_ISDIR(fsp->base_fsp->fsp_name->st.st_ex_mode))
    1562             :         {
    1563             :                 /* sorry, but directories don't have a resource fork */
    1564           0 :                 errno = ENOENT;
    1565           0 :                 rc = -1;
    1566           0 :                 goto exit;
    1567             :         }
    1568             : 
    1569             :         /*
    1570             :          * We return a fake_fd to the vfs modules above,
    1571             :          * while we open an internal backend fsp for the
    1572             :          * '._' file for the next vfs modules.
    1573             :          *
    1574             :          * Note that adouble_open_from_base_fsp() recurses
    1575             :          * into fruit_openat(), but it'll just pass to
    1576             :          * the next module as just opens a flat file on
    1577             :          * disk.
    1578             :          */
    1579             : 
    1580        1012 :         fd = vfs_fake_fd();
    1581        1012 :         if (fd == -1) {
    1582           0 :                 rc = fd;
    1583           0 :                 goto exit;
    1584             :         }
    1585             : 
    1586        1012 :         status = adouble_open_from_base_fsp(fsp->conn->cwd_fsp,
    1587             :                                             fsp->base_fsp,
    1588             :                                             ADOUBLE_RSRC,
    1589             :                                             flags,
    1590             :                                             mode,
    1591             :                                             &ad_fsp);
    1592        1012 :         if (!NT_STATUS_IS_OK(status)) {
    1593         190 :                 errno = map_errno_from_nt_status(status);
    1594         190 :                 rc = -1;
    1595         190 :                 goto exit;
    1596             :         }
    1597             : 
    1598             :         /*
    1599             :          * Now we need to glue both handles together,
    1600             :          * so that they automatically detach each other
    1601             :          * on close.
    1602             :          */
    1603         822 :         fio = fruit_get_complete_fio(handle, fsp);
    1604         822 :         if (fio == NULL) {
    1605           0 :                 DBG_ERR("fio=NULL for [%s]\n", fsp_str_dbg(fsp));
    1606           0 :                 errno = EBADF;
    1607           0 :                 rc = -1;
    1608           0 :                 goto exit;
    1609             :         }
    1610             : 
    1611         822 :         ref_fio = VFS_ADD_FSP_EXTENSION(handle, ad_fsp,
    1612             :                                         struct fio,
    1613             :                                         fio_ref_destroy_fn);
    1614         822 :         if (ref_fio == NULL) {
    1615           0 :                 int saved_errno = errno;
    1616           0 :                 fd_close(ad_fsp);
    1617           0 :                 file_free(NULL, ad_fsp);
    1618           0 :                 ad_fsp = NULL;
    1619           0 :                 errno = saved_errno;
    1620           0 :                 rc = -1;
    1621           0 :                 goto exit;
    1622             :         }
    1623             : 
    1624         822 :         SMB_ASSERT(ref_fio->fsp == NULL);
    1625         822 :         ref_fio->handle = handle;
    1626         822 :         ref_fio->fsp = ad_fsp;
    1627         822 :         ref_fio->type = ADOUBLE_RSRC;
    1628         822 :         ref_fio->config = config;
    1629         822 :         ref_fio->real_fio = fio;
    1630         822 :         SMB_ASSERT(fio->ad_fsp == NULL);
    1631         822 :         fio->ad_fsp = ad_fsp;
    1632         822 :         fio->fake_fd = true;
    1633             : 
    1634        1012 : exit:
    1635             : 
    1636        1012 :         DEBUG(10, ("fruit_open resource fork: rc=%d\n", rc));
    1637        1012 :         if (rc != 0) {
    1638         190 :                 int saved_errno = errno;
    1639         190 :                 if (fd != -1) {
    1640         190 :                         vfs_fake_fd_close(fd);
    1641             :                 }
    1642         190 :                 errno = saved_errno;
    1643         190 :                 return rc;
    1644             :         }
    1645         822 :         return fd;
    1646             : }
    1647             : 
    1648           0 : static int fruit_open_rsrc_xattr(vfs_handle_struct *handle,
    1649             :                                  const struct files_struct *dirfsp,
    1650             :                                  const struct smb_filename *smb_fname,
    1651             :                                  files_struct *fsp,
    1652             :                                  int flags,
    1653             :                                  mode_t mode)
    1654             : {
    1655             : #ifdef HAVE_ATTROPEN
    1656             :         int fd = -1;
    1657             : 
    1658             :         /*
    1659             :          * As there's no attropenat() this is only going to work with AT_FDCWD.
    1660             :          */
    1661             :         SMB_ASSERT(fsp_get_pathref_fd(dirfsp) == AT_FDCWD);
    1662             : 
    1663             :         fd = attropen(smb_fname->base_name,
    1664             :                       AFPRESOURCE_EA_NETATALK,
    1665             :                       flags,
    1666             :                       mode);
    1667             :         if (fd == -1) {
    1668             :                 return -1;
    1669             :         }
    1670             : 
    1671             :         return fd;
    1672             : 
    1673             : #else
    1674           0 :         errno = ENOSYS;
    1675           0 :         return -1;
    1676             : #endif
    1677             : }
    1678             : 
    1679        1344 : static int fruit_open_rsrc(vfs_handle_struct *handle,
    1680             :                            const struct files_struct *dirfsp,
    1681             :                            const struct smb_filename *smb_fname,
    1682             :                            files_struct *fsp, int flags, mode_t mode)
    1683             : {
    1684             :         int fd;
    1685        1344 :         struct fruit_config_data *config = NULL;
    1686        1344 :         struct fio *fio = NULL;
    1687             : 
    1688        1344 :         DBG_DEBUG("Path [%s]\n", smb_fname_str_dbg(smb_fname));
    1689             : 
    1690        1344 :         SMB_VFS_HANDLE_GET_DATA(handle, config,
    1691             :                                 struct fruit_config_data, return -1);
    1692             : 
    1693        1344 :         fio = VFS_ADD_FSP_EXTENSION(handle, fsp, struct fio, fio_destroy_fn);
    1694        1344 :         fio->handle = handle;
    1695        1344 :         fio->fsp = fsp;
    1696        1344 :         fio->type = ADOUBLE_RSRC;
    1697        1344 :         fio->config = config;
    1698             : 
    1699        1344 :         switch (config->rsrc) {
    1700         332 :         case FRUIT_RSRC_STREAM: {
    1701         332 :                 struct vfs_open_how how = {
    1702             :                         .flags = flags, .mode = mode,
    1703             :                 };
    1704         332 :                 fd = SMB_VFS_NEXT_OPENAT(handle,
    1705             :                                          dirfsp,
    1706             :                                          smb_fname,
    1707             :                                          fsp,
    1708             :                                          &how);
    1709         332 :                 break;
    1710             :         }
    1711             : 
    1712        1012 :         case FRUIT_RSRC_ADFILE:
    1713        1012 :                 fd = fruit_open_rsrc_adouble(handle, dirfsp, smb_fname,
    1714             :                                              fsp, flags, mode);
    1715        1012 :                 break;
    1716             : 
    1717           0 :         case FRUIT_RSRC_XATTR:
    1718           0 :                 fd = fruit_open_rsrc_xattr(handle, dirfsp, smb_fname,
    1719             :                                            fsp, flags, mode);
    1720           0 :                 break;
    1721             : 
    1722           0 :         default:
    1723           0 :                 DBG_ERR("Unexpected rsrc config [%d]\n", config->rsrc);
    1724           0 :                 errno = EINVAL;
    1725           0 :                 return -1;
    1726             :         }
    1727             : 
    1728        1344 :         DBG_DEBUG("Path [%s] fd [%d]\n", smb_fname_str_dbg(smb_fname), fd);
    1729             : 
    1730        1344 :         if (fd == -1) {
    1731         244 :                 return -1;
    1732             :         }
    1733             : 
    1734        1100 :         return fd;
    1735             : }
    1736             : 
    1737       95682 : static int fruit_openat(vfs_handle_struct *handle,
    1738             :                         const struct files_struct *dirfsp,
    1739             :                         const struct smb_filename *smb_fname,
    1740             :                         files_struct *fsp,
    1741             :                         const struct vfs_open_how *how)
    1742             : {
    1743             :         int fd;
    1744             : 
    1745       95682 :         DBG_DEBUG("Path [%s]\n", smb_fname_str_dbg(smb_fname));
    1746             : 
    1747       95682 :         if (!is_named_stream(smb_fname)) {
    1748       87960 :                 return SMB_VFS_NEXT_OPENAT(handle,
    1749             :                                            dirfsp,
    1750             :                                            smb_fname,
    1751             :                                            fsp,
    1752             :                                            how);
    1753             :         }
    1754             : 
    1755        7722 :         if (how->resolve != 0) {
    1756           0 :                 errno = ENOSYS;
    1757           0 :                 return -1;
    1758             :         }
    1759             : 
    1760        7722 :         SMB_ASSERT(fsp_is_alternate_stream(fsp));
    1761             : 
    1762        7722 :         if (is_afpinfo_stream(smb_fname->stream_name)) {
    1763        5304 :                 fd = fruit_open_meta(handle,
    1764             :                                      dirfsp,
    1765             :                                      smb_fname,
    1766             :                                      fsp,
    1767        5304 :                                      how->flags,
    1768        5304 :                                      how->mode);
    1769        2418 :         } else if (is_afpresource_stream(smb_fname->stream_name)) {
    1770        1344 :                 fd = fruit_open_rsrc(handle,
    1771             :                                      dirfsp,
    1772             :                                      smb_fname,
    1773             :                                      fsp,
    1774        1344 :                                      how->flags,
    1775        1344 :                                      how->mode);
    1776             :         } else {
    1777        1074 :                 fd = SMB_VFS_NEXT_OPENAT(handle,
    1778             :                                          dirfsp,
    1779             :                                          smb_fname,
    1780             :                                          fsp,
    1781             :                                          how);
    1782             :         }
    1783             : 
    1784        7722 :         DBG_DEBUG("Path [%s] fd [%d]\n", smb_fname_str_dbg(smb_fname), fd);
    1785             : 
    1786             :         /* Prevent reopen optimisation */
    1787        7722 :         fsp->fsp_flags.have_proc_fds = false;
    1788        7722 :         return fd;
    1789             : }
    1790             : 
    1791        3420 : static int fruit_close_meta(vfs_handle_struct *handle,
    1792             :                             files_struct *fsp)
    1793             : {
    1794             :         int ret;
    1795        3420 :         struct fruit_config_data *config = NULL;
    1796             : 
    1797        3420 :         SMB_VFS_HANDLE_GET_DATA(handle, config,
    1798             :                                 struct fruit_config_data, return -1);
    1799             : 
    1800        3420 :         switch (config->meta) {
    1801        2548 :         case FRUIT_META_STREAM:
    1802             :         {
    1803        2548 :                 struct fio *fio = fruit_get_complete_fio(handle, fsp);
    1804        2548 :                 if (fio == NULL) {
    1805           0 :                         return -1;
    1806             :                 }
    1807        2548 :                 if (fio->fake_fd) {
    1808         738 :                         ret = vfs_fake_fd_close(fsp_get_pathref_fd(fsp));
    1809         738 :                         fsp_set_fd(fsp, -1);
    1810             :                 } else {
    1811        1810 :                         ret = SMB_VFS_NEXT_CLOSE(handle, fsp);
    1812             :                 }
    1813        2548 :                 break;
    1814             :         }
    1815         872 :         case FRUIT_META_NETATALK:
    1816         872 :                 ret = vfs_fake_fd_close(fsp_get_pathref_fd(fsp));
    1817         872 :                 fsp_set_fd(fsp, -1);
    1818         872 :                 break;
    1819             : 
    1820           0 :         default:
    1821           0 :                 DBG_ERR("Unexpected meta config [%d]\n", config->meta);
    1822           0 :                 return -1;
    1823             :         }
    1824             : 
    1825        3420 :         return ret;
    1826             : }
    1827             : 
    1828             : 
    1829        1084 : static int fruit_close_rsrc(vfs_handle_struct *handle,
    1830             :                             files_struct *fsp)
    1831             : {
    1832             :         int ret;
    1833        1084 :         struct fruit_config_data *config = NULL;
    1834             : 
    1835        1084 :         SMB_VFS_HANDLE_GET_DATA(handle, config,
    1836             :                                 struct fruit_config_data, return -1);
    1837             : 
    1838        1084 :         switch (config->rsrc) {
    1839         274 :         case FRUIT_RSRC_STREAM:
    1840         274 :                 ret = SMB_VFS_NEXT_CLOSE(handle, fsp);
    1841         274 :                 break;
    1842             : 
    1843         810 :         case FRUIT_RSRC_ADFILE:
    1844             :         {
    1845         810 :                 struct fio *fio = fruit_get_complete_fio(handle, fsp);
    1846         810 :                 if (fio == NULL) {
    1847           0 :                         return -1;
    1848             :                 }
    1849         810 :                 fio_close_ad_fsp(fio);
    1850         810 :                 ret = vfs_fake_fd_close(fsp_get_pathref_fd(fsp));
    1851         810 :                 fsp_set_fd(fsp, -1);
    1852         810 :                 break;
    1853             :         }
    1854             : 
    1855           0 :         case FRUIT_RSRC_XATTR:
    1856           0 :                 ret = vfs_fake_fd_close(fsp_get_pathref_fd(fsp));
    1857           0 :                 fsp_set_fd(fsp, -1);
    1858           0 :                 break;
    1859             : 
    1860           0 :         default:
    1861           0 :                 DBG_ERR("Unexpected rsrc config [%d]\n", config->rsrc);
    1862           0 :                 return -1;
    1863             :         }
    1864             : 
    1865        1084 :         return ret;
    1866             : }
    1867             : 
    1868       64420 : static int fruit_close(vfs_handle_struct *handle,
    1869             :                        files_struct *fsp)
    1870             : {
    1871             :         int ret;
    1872             :         int fd;
    1873             : 
    1874       64420 :         fd = fsp_get_pathref_fd(fsp);
    1875             : 
    1876       64420 :         DBG_DEBUG("Path [%s] fd [%d]\n", smb_fname_str_dbg(fsp->fsp_name), fd);
    1877             : 
    1878       64420 :         if (!fsp_is_alternate_stream(fsp)) {
    1879       59004 :                 return SMB_VFS_NEXT_CLOSE(handle, fsp);
    1880             :         }
    1881             : 
    1882        5416 :         if (is_afpinfo_stream(fsp->fsp_name->stream_name)) {
    1883        3420 :                 ret = fruit_close_meta(handle, fsp);
    1884        1996 :         } else if (is_afpresource_stream(fsp->fsp_name->stream_name)) {
    1885        1084 :                 ret = fruit_close_rsrc(handle, fsp);
    1886             :         } else {
    1887         912 :                 ret = SMB_VFS_NEXT_CLOSE(handle, fsp);
    1888             :         }
    1889             : 
    1890        5416 :         return ret;
    1891             : }
    1892             : 
    1893           8 : static int fruit_renameat(struct vfs_handle_struct *handle,
    1894             :                         files_struct *srcfsp,
    1895             :                         const struct smb_filename *smb_fname_src,
    1896             :                         files_struct *dstfsp,
    1897             :                         const struct smb_filename *smb_fname_dst)
    1898             : {
    1899           8 :         int rc = -1;
    1900           8 :         struct fruit_config_data *config = NULL;
    1901           8 :         struct smb_filename *src_adp_smb_fname = NULL;
    1902           8 :         struct smb_filename *dst_adp_smb_fname = NULL;
    1903             : 
    1904           8 :         SMB_VFS_HANDLE_GET_DATA(handle, config,
    1905             :                                 struct fruit_config_data, return -1);
    1906             : 
    1907           8 :         if (!VALID_STAT(smb_fname_src->st)) {
    1908           0 :                 DBG_ERR("Need valid stat for [%s]\n",
    1909             :                         smb_fname_str_dbg(smb_fname_src));
    1910           0 :                 return -1;
    1911             :         }
    1912             : 
    1913           8 :         rc = SMB_VFS_NEXT_RENAMEAT(handle,
    1914             :                                 srcfsp,
    1915             :                                 smb_fname_src,
    1916             :                                 dstfsp,
    1917             :                                 smb_fname_dst);
    1918           8 :         if (rc != 0) {
    1919           0 :                 return -1;
    1920             :         }
    1921             : 
    1922           8 :         if ((config->rsrc != FRUIT_RSRC_ADFILE) ||
    1923           6 :             (!S_ISREG(smb_fname_src->st.st_ex_mode)))
    1924             :         {
    1925           8 :                 return 0;
    1926             :         }
    1927             : 
    1928           0 :         rc = adouble_path(talloc_tos(), smb_fname_src, &src_adp_smb_fname);
    1929           0 :         if (rc != 0) {
    1930           0 :                 goto done;
    1931             :         }
    1932             : 
    1933           0 :         rc = adouble_path(talloc_tos(), smb_fname_dst, &dst_adp_smb_fname);
    1934           0 :         if (rc != 0) {
    1935           0 :                 goto done;
    1936             :         }
    1937             : 
    1938           0 :         DBG_DEBUG("%s -> %s\n",
    1939             :                   smb_fname_str_dbg(src_adp_smb_fname),
    1940             :                   smb_fname_str_dbg(dst_adp_smb_fname));
    1941             : 
    1942           0 :         rc = SMB_VFS_NEXT_RENAMEAT(handle,
    1943             :                         srcfsp,
    1944             :                         src_adp_smb_fname,
    1945             :                         dstfsp,
    1946             :                         dst_adp_smb_fname);
    1947           0 :         if (errno == ENOENT) {
    1948           0 :                 rc = 0;
    1949             :         }
    1950             : 
    1951           0 : done:
    1952           0 :         TALLOC_FREE(src_adp_smb_fname);
    1953           0 :         TALLOC_FREE(dst_adp_smb_fname);
    1954           0 :         return rc;
    1955             : }
    1956             : 
    1957         360 : static int fruit_unlink_meta_stream(vfs_handle_struct *handle,
    1958             :                                 struct files_struct *dirfsp,
    1959             :                                 const struct smb_filename *smb_fname)
    1960             : {
    1961         360 :         return SMB_VFS_NEXT_UNLINKAT(handle,
    1962             :                                 dirfsp,
    1963             :                                 smb_fname,
    1964             :                                 0);
    1965             : }
    1966             : 
    1967         124 : static int fruit_unlink_meta_netatalk(vfs_handle_struct *handle,
    1968             :                                       const struct smb_filename *smb_fname)
    1969             : {
    1970         124 :         SMB_ASSERT(smb_fname->fsp != NULL);
    1971         124 :         SMB_ASSERT(fsp_is_alternate_stream(smb_fname->fsp));
    1972         124 :         return SMB_VFS_FREMOVEXATTR(smb_fname->fsp->base_fsp,
    1973             :                                    AFPINFO_EA_NETATALK);
    1974             : }
    1975             : 
    1976         484 : static int fruit_unlink_meta(vfs_handle_struct *handle,
    1977             :                         struct files_struct *dirfsp,
    1978             :                         const struct smb_filename *smb_fname)
    1979             : {
    1980         484 :         struct fruit_config_data *config = NULL;
    1981             :         int rc;
    1982             : 
    1983         484 :         SMB_VFS_HANDLE_GET_DATA(handle, config,
    1984             :                                 struct fruit_config_data, return -1);
    1985             : 
    1986         484 :         switch (config->meta) {
    1987         360 :         case FRUIT_META_STREAM:
    1988         360 :                 rc = fruit_unlink_meta_stream(handle,
    1989             :                                 dirfsp,
    1990             :                                 smb_fname);
    1991         360 :                 break;
    1992             : 
    1993         124 :         case FRUIT_META_NETATALK:
    1994         124 :                 rc = fruit_unlink_meta_netatalk(handle, smb_fname);
    1995         124 :                 break;
    1996             : 
    1997           0 :         default:
    1998           0 :                 DBG_ERR("Unsupported meta config [%d]\n", config->meta);
    1999           0 :                 return -1;
    2000             :         }
    2001             : 
    2002         484 :         return rc;
    2003             : }
    2004             : 
    2005         162 : static int fruit_unlink_rsrc_stream(vfs_handle_struct *handle,
    2006             :                                 struct files_struct *dirfsp,
    2007             :                                 const struct smb_filename *smb_fname,
    2008             :                                 bool force_unlink)
    2009             : {
    2010             :         int ret;
    2011             : 
    2012         162 :         if (!force_unlink) {
    2013          38 :                 struct smb_filename *full_fname = NULL;
    2014             :                 off_t size;
    2015             : 
    2016             :                 /*
    2017             :                  * TODO: use SMB_VFS_STATX() once we have it.
    2018             :                  */
    2019             : 
    2020          38 :                 full_fname = full_path_from_dirfsp_atname(talloc_tos(),
    2021             :                                                           dirfsp,
    2022             :                                                           smb_fname);
    2023          38 :                 if (full_fname == NULL) {
    2024           0 :                         return -1;
    2025             :                 }
    2026             : 
    2027             :                 /*
    2028             :                  * 0 byte resource fork streams are not listed by
    2029             :                  * vfs_streaminfo, as a result stream cleanup/deletion of file
    2030             :                  * deletion doesn't remove the resourcefork stream.
    2031             :                  */
    2032             : 
    2033          38 :                 ret = SMB_VFS_NEXT_STAT(handle, full_fname);
    2034          38 :                 if (ret != 0) {
    2035           0 :                         TALLOC_FREE(full_fname);
    2036           0 :                         DBG_ERR("stat [%s] failed [%s]\n",
    2037             :                                 smb_fname_str_dbg(full_fname), strerror(errno));
    2038           0 :                         return -1;
    2039             :                 }
    2040             : 
    2041          38 :                 size = full_fname->st.st_ex_size;
    2042          38 :                 TALLOC_FREE(full_fname);
    2043             : 
    2044          38 :                 if (size > 0) {
    2045             :                         /* OS X ignores resource fork stream delete requests */
    2046          38 :                         return 0;
    2047             :                 }
    2048             :         }
    2049             : 
    2050         124 :         ret = SMB_VFS_NEXT_UNLINKAT(handle,
    2051             :                         dirfsp,
    2052             :                         smb_fname,
    2053             :                         0);
    2054         124 :         if ((ret != 0) && (errno == ENOENT) && force_unlink) {
    2055          80 :                 ret = 0;
    2056             :         }
    2057             : 
    2058         124 :         return ret;
    2059             : }
    2060             : 
    2061         700 : static int fruit_unlink_rsrc_adouble(vfs_handle_struct *handle,
    2062             :                                 struct files_struct *dirfsp,
    2063             :                                 const struct smb_filename *smb_fname,
    2064             :                                 bool force_unlink)
    2065             : {
    2066             :         int rc;
    2067         700 :         struct adouble *ad = NULL;
    2068         700 :         struct smb_filename *adp_smb_fname = NULL;
    2069             : 
    2070         700 :         if (!force_unlink) {
    2071         120 :                 struct smb_filename *full_fname = NULL;
    2072             : 
    2073         120 :                 full_fname = full_path_from_dirfsp_atname(talloc_tos(),
    2074             :                                                           dirfsp,
    2075             :                                                           smb_fname);
    2076         120 :                 if (full_fname == NULL) {
    2077           0 :                         return -1;
    2078             :                 }
    2079             : 
    2080         120 :                 ad = ad_get(talloc_tos(), handle, full_fname,
    2081             :                             ADOUBLE_RSRC);
    2082         120 :                 TALLOC_FREE(full_fname);
    2083         120 :                 if (ad == NULL) {
    2084           0 :                         errno = ENOENT;
    2085           0 :                         return -1;
    2086             :                 }
    2087             : 
    2088             : 
    2089             :                 /*
    2090             :                  * 0 byte resource fork streams are not listed by
    2091             :                  * vfs_streaminfo, as a result stream cleanup/deletion of file
    2092             :                  * deletion doesn't remove the resourcefork stream.
    2093             :                  */
    2094             : 
    2095         120 :                 if (ad_getentrylen(ad, ADEID_RFORK) > 0) {
    2096             :                         /* OS X ignores resource fork stream delete requests */
    2097         120 :                         TALLOC_FREE(ad);
    2098         120 :                         return 0;
    2099             :                 }
    2100             : 
    2101           0 :                 TALLOC_FREE(ad);
    2102             :         }
    2103             : 
    2104         580 :         rc = adouble_path(talloc_tos(), smb_fname, &adp_smb_fname);
    2105         580 :         if (rc != 0) {
    2106           0 :                 return -1;
    2107             :         }
    2108             : 
    2109         580 :         rc = SMB_VFS_NEXT_UNLINKAT(handle,
    2110             :                         dirfsp,
    2111             :                         adp_smb_fname,
    2112             :                         0);
    2113         580 :         TALLOC_FREE(adp_smb_fname);
    2114         580 :         if ((rc != 0) && (errno == ENOENT || errno == ENAMETOOLONG) && force_unlink) {
    2115         452 :                 rc = 0;
    2116             :         }
    2117             : 
    2118         580 :         return rc;
    2119             : }
    2120             : 
    2121           0 : static int fruit_unlink_rsrc_xattr(vfs_handle_struct *handle,
    2122             :                                    const struct smb_filename *smb_fname,
    2123             :                                    bool force_unlink)
    2124             : {
    2125             :         /*
    2126             :          * OS X ignores resource fork stream delete requests, so nothing to do
    2127             :          * here. Removing the file will remove the xattr anyway, so we don't
    2128             :          * have to take care of removing 0 byte resource forks that could be
    2129             :          * left behind.
    2130             :          */
    2131           0 :         return 0;
    2132             : }
    2133             : 
    2134         862 : static int fruit_unlink_rsrc(vfs_handle_struct *handle,
    2135             :                         struct files_struct *dirfsp,
    2136             :                         const struct smb_filename *smb_fname,
    2137             :                         bool force_unlink)
    2138             : {
    2139         862 :         struct fruit_config_data *config = NULL;
    2140             :         int rc;
    2141             : 
    2142         862 :         SMB_VFS_HANDLE_GET_DATA(handle, config,
    2143             :                                 struct fruit_config_data, return -1);
    2144             : 
    2145         862 :         switch (config->rsrc) {
    2146         162 :         case FRUIT_RSRC_STREAM:
    2147         162 :                 rc = fruit_unlink_rsrc_stream(handle,
    2148             :                                 dirfsp,
    2149             :                                 smb_fname,
    2150             :                                 force_unlink);
    2151         162 :                 break;
    2152             : 
    2153         700 :         case FRUIT_RSRC_ADFILE:
    2154         700 :                 rc = fruit_unlink_rsrc_adouble(handle,
    2155             :                                 dirfsp,
    2156             :                                 smb_fname,
    2157             :                                 force_unlink);
    2158         700 :                 break;
    2159             : 
    2160           0 :         case FRUIT_RSRC_XATTR:
    2161           0 :                 rc = fruit_unlink_rsrc_xattr(handle, smb_fname, force_unlink);
    2162           0 :                 break;
    2163             : 
    2164           0 :         default:
    2165           0 :                 DBG_ERR("Unsupported rsrc config [%d]\n", config->rsrc);
    2166           0 :                 return -1;
    2167             :         }
    2168             : 
    2169         862 :         return rc;
    2170             : }
    2171             : 
    2172           8 : static int fruit_fchmod(vfs_handle_struct *handle,
    2173             :                       struct files_struct *fsp,
    2174             :                       mode_t mode)
    2175             : {
    2176           8 :         int rc = -1;
    2177           8 :         struct fruit_config_data *config = NULL;
    2178           8 :         struct smb_filename *smb_fname_adp = NULL;
    2179           8 :         const struct smb_filename *smb_fname = NULL;
    2180             :         NTSTATUS status;
    2181             : 
    2182           8 :         rc = SMB_VFS_NEXT_FCHMOD(handle, fsp, mode);
    2183           8 :         if (rc != 0) {
    2184           0 :                 return rc;
    2185             :         }
    2186             : 
    2187           8 :         smb_fname = fsp->fsp_name;
    2188           8 :         SMB_VFS_HANDLE_GET_DATA(handle, config,
    2189             :                                 struct fruit_config_data, return -1);
    2190             : 
    2191           8 :         if (config->rsrc != FRUIT_RSRC_ADFILE) {
    2192           2 :                 return 0;
    2193             :         }
    2194             : 
    2195           6 :         if (!VALID_STAT(smb_fname->st)) {
    2196           0 :                 return 0;
    2197             :         }
    2198             : 
    2199           6 :         if (!S_ISREG(smb_fname->st.st_ex_mode)) {
    2200           0 :                 return 0;
    2201             :         }
    2202             : 
    2203           6 :         rc = adouble_path(talloc_tos(), smb_fname, &smb_fname_adp);
    2204           6 :         if (rc != 0) {
    2205           0 :                 return -1;
    2206             :         }
    2207             : 
    2208           6 :         status = openat_pathref_fsp(handle->conn->cwd_fsp,
    2209             :                                     smb_fname_adp);
    2210           6 :         if (!NT_STATUS_IS_OK(status)) {
    2211             :                 /* detect ENOENT (mapped to OBJECT_NAME_NOT_FOUND) */
    2212           6 :                 if (NT_STATUS_EQUAL(status,
    2213             :                                     NT_STATUS_OBJECT_NAME_NOT_FOUND)){
    2214           6 :                         rc = 0;
    2215           6 :                         goto out;
    2216             :                 }
    2217           0 :                 rc = -1;
    2218           0 :                 goto out;
    2219             :         }
    2220             : 
    2221           0 :         DBG_DEBUG("%s\n", smb_fname_adp->base_name);
    2222             : 
    2223           0 :         rc = SMB_VFS_NEXT_FCHMOD(handle, smb_fname_adp->fsp, mode);
    2224           0 :         if (errno == ENOENT) {
    2225           0 :                 rc = 0;
    2226             :         }
    2227           0 : out:
    2228           6 :         TALLOC_FREE(smb_fname_adp);
    2229           6 :         return rc;
    2230             : }
    2231             : 
    2232        1896 : static int fruit_unlinkat(vfs_handle_struct *handle,
    2233             :                         struct files_struct *dirfsp,
    2234             :                         const struct smb_filename *smb_fname,
    2235             :                         int flags)
    2236             : {
    2237        1896 :         struct fruit_config_data *config = NULL;
    2238        1896 :         struct smb_filename *rsrc_smb_fname = NULL;
    2239             :         int ret;
    2240             : 
    2241        1896 :         if (flags & AT_REMOVEDIR) {
    2242         422 :                 return SMB_VFS_NEXT_UNLINKAT(handle,
    2243             :                                              dirfsp,
    2244             :                                              smb_fname,
    2245             :                                              AT_REMOVEDIR);
    2246             :         }
    2247             : 
    2248        1474 :         SMB_VFS_HANDLE_GET_DATA(handle, config,
    2249             :                                 struct fruit_config_data, return -1);
    2250             : 
    2251        1474 :         if (is_afpinfo_stream(smb_fname->stream_name)) {
    2252         484 :                 return fruit_unlink_meta(handle,
    2253             :                                 dirfsp,
    2254             :                                 smb_fname);
    2255         990 :         } else if (is_afpresource_stream(smb_fname->stream_name)) {
    2256         158 :                 return fruit_unlink_rsrc(handle,
    2257             :                                 dirfsp,
    2258             :                                 smb_fname,
    2259             :                                 false);
    2260         832 :         } else if (is_named_stream(smb_fname)) {
    2261         110 :                 return SMB_VFS_NEXT_UNLINKAT(handle,
    2262             :                                 dirfsp,
    2263             :                                 smb_fname,
    2264             :                                 0);
    2265         722 :         } else if (is_adouble_file(smb_fname->base_name)) {
    2266          18 :                 return SMB_VFS_NEXT_UNLINKAT(handle,
    2267             :                                 dirfsp,
    2268             :                                 smb_fname,
    2269             :                                 0);
    2270             :         }
    2271             : 
    2272             :         /*
    2273             :          * A request to delete the base file. Because 0 byte resource
    2274             :          * fork streams are not listed by fruit_streaminfo,
    2275             :          * delete_all_streams() can't remove 0 byte resource fork
    2276             :          * streams, so we have to cleanup this here.
    2277             :          */
    2278         704 :         rsrc_smb_fname = synthetic_smb_fname(talloc_tos(),
    2279         704 :                                              smb_fname->base_name,
    2280             :                                              AFPRESOURCE_STREAM_NAME,
    2281             :                                              NULL,
    2282         704 :                                              smb_fname->twrp,
    2283         704 :                                              smb_fname->flags);
    2284         704 :         if (rsrc_smb_fname == NULL) {
    2285           0 :                 return -1;
    2286             :         }
    2287             : 
    2288         704 :         ret = fruit_unlink_rsrc(handle, dirfsp, rsrc_smb_fname, true);
    2289         704 :         if ((ret != 0) && (errno != ENOENT)) {
    2290           0 :                 DBG_ERR("Forced unlink of [%s] failed [%s]\n",
    2291             :                         smb_fname_str_dbg(rsrc_smb_fname), strerror(errno));
    2292           0 :                 TALLOC_FREE(rsrc_smb_fname);
    2293           0 :                 return -1;
    2294             :         }
    2295         704 :         TALLOC_FREE(rsrc_smb_fname);
    2296             : 
    2297         704 :         return SMB_VFS_NEXT_UNLINKAT(handle,
    2298             :                         dirfsp,
    2299             :                         smb_fname,
    2300             :                         0);
    2301             : }
    2302             : 
    2303         404 : static ssize_t fruit_pread_meta_stream(vfs_handle_struct *handle,
    2304             :                                        files_struct *fsp, void *data,
    2305             :                                        size_t n, off_t offset)
    2306             : {
    2307         404 :         struct fio *fio = fruit_get_complete_fio(handle, fsp);
    2308             :         ssize_t nread;
    2309             :         int ret;
    2310             : 
    2311         404 :         if ((fio == NULL) || fio->fake_fd) {
    2312          12 :                 return -1;
    2313             :         }
    2314             : 
    2315         392 :         nread = SMB_VFS_NEXT_PREAD(handle, fsp, data, n, offset);
    2316         392 :         if (nread == -1 || nread == n) {
    2317         392 :                 return nread;
    2318             :         }
    2319             : 
    2320           0 :         DBG_ERR("Removing [%s] after short read [%zd]\n",
    2321             :                 fsp_str_dbg(fsp), nread);
    2322             : 
    2323           0 :         ret = SMB_VFS_NEXT_UNLINKAT(handle,
    2324             :                         fsp->conn->cwd_fsp,
    2325             :                         fsp->fsp_name,
    2326             :                         0);
    2327           0 :         if (ret != 0) {
    2328           0 :                 DBG_ERR("Removing [%s] failed\n", fsp_str_dbg(fsp));
    2329           0 :                 return -1;
    2330             :         }
    2331             : 
    2332           0 :         errno = EINVAL;
    2333           0 :         return -1;
    2334             : }
    2335             : 
    2336         138 : static ssize_t fruit_pread_meta_adouble(vfs_handle_struct *handle,
    2337             :                                         files_struct *fsp, void *data,
    2338             :                                         size_t n, off_t offset)
    2339             : {
    2340         138 :         AfpInfo *ai = NULL;
    2341         138 :         struct adouble *ad = NULL;
    2342             :         char afpinfo_buf[AFP_INFO_SIZE];
    2343         138 :         char *p = NULL;
    2344             :         ssize_t nread;
    2345             : 
    2346         138 :         ai = afpinfo_new(talloc_tos());
    2347         138 :         if (ai == NULL) {
    2348           0 :                 return -1;
    2349             :         }
    2350             : 
    2351         138 :         ad = ad_fget(talloc_tos(), handle, fsp, ADOUBLE_META);
    2352         138 :         if (ad == NULL) {
    2353           4 :                 nread = -1;
    2354           4 :                 goto fail;
    2355             :         }
    2356             : 
    2357         134 :         p = ad_get_entry(ad, ADEID_FINDERI);
    2358         134 :         if (p == NULL) {
    2359           0 :                 DBG_ERR("No ADEID_FINDERI for [%s]\n", fsp_str_dbg(fsp));
    2360           0 :                 nread = -1;
    2361           0 :                 goto fail;
    2362             :         }
    2363             : 
    2364         134 :         memcpy(&ai->afpi_FinderInfo[0], p, ADEDLEN_FINDERI);
    2365             : 
    2366         134 :         nread = afpinfo_pack(ai, afpinfo_buf);
    2367         134 :         if (nread != AFP_INFO_SIZE) {
    2368           0 :                 nread = -1;
    2369           0 :                 goto fail;
    2370             :         }
    2371             : 
    2372         134 :         memcpy(data, afpinfo_buf, n);
    2373         134 :         nread = n;
    2374             : 
    2375         138 : fail:
    2376         138 :         TALLOC_FREE(ai);
    2377         138 :         return nread;
    2378             : }
    2379             : 
    2380         552 : static ssize_t fruit_pread_meta(vfs_handle_struct *handle,
    2381             :                                 files_struct *fsp, void *data,
    2382             :                                 size_t n, off_t offset)
    2383             : {
    2384         552 :         struct fio *fio = fruit_get_complete_fio(handle, fsp);
    2385             :         ssize_t nread;
    2386             :         ssize_t to_return;
    2387             : 
    2388             :         /*
    2389             :          * OS X has a off-by-1 error in the offset calculation, so we're
    2390             :          * bug compatible here. It won't hurt, as any relevant real
    2391             :          * world read requests from the AFP_AfpInfo stream will be
    2392             :          * offset=0 n=60. offset is ignored anyway, see below.
    2393             :          */
    2394         552 :         if ((offset < 0) || (offset >= AFP_INFO_SIZE + 1)) {
    2395          10 :                 return 0;
    2396             :         }
    2397             : 
    2398         542 :         if (fio == NULL) {
    2399           0 :                 DBG_ERR("Failed to fetch fsp extension\n");
    2400           0 :                 return -1;
    2401             :         }
    2402             : 
    2403             :         /* Yes, macOS always reads from offset 0 */
    2404         542 :         offset = 0;
    2405         542 :         to_return = MIN(n, AFP_INFO_SIZE);
    2406             : 
    2407         542 :         switch (fio->config->meta) {
    2408         404 :         case FRUIT_META_STREAM:
    2409         404 :                 nread = fruit_pread_meta_stream(handle, fsp, data,
    2410             :                                                 to_return, offset);
    2411         404 :                 break;
    2412             : 
    2413         138 :         case FRUIT_META_NETATALK:
    2414         138 :                 nread = fruit_pread_meta_adouble(handle, fsp, data,
    2415             :                                                  to_return, offset);
    2416         138 :                 break;
    2417             : 
    2418           0 :         default:
    2419           0 :                 DBG_ERR("Unexpected meta config [%d]\n", fio->config->meta);
    2420           0 :                 return -1;
    2421             :         }
    2422             : 
    2423         542 :         if (nread == -1 && fio->fake_fd) {
    2424          16 :                 AfpInfo *ai = NULL;
    2425             :                 char afpinfo_buf[AFP_INFO_SIZE];
    2426             : 
    2427          16 :                 ai = afpinfo_new(talloc_tos());
    2428          16 :                 if (ai == NULL) {
    2429           0 :                         return -1;
    2430             :                 }
    2431             : 
    2432          16 :                 nread = afpinfo_pack(ai, afpinfo_buf);
    2433          16 :                 TALLOC_FREE(ai);
    2434          16 :                 if (nread != AFP_INFO_SIZE) {
    2435           0 :                         return -1;
    2436             :                 }
    2437             : 
    2438          16 :                 memcpy(data, afpinfo_buf, to_return);
    2439          16 :                 return to_return;
    2440             :         }
    2441             : 
    2442         526 :         return nread;
    2443             : }
    2444             : 
    2445          20 : static ssize_t fruit_pread_rsrc_stream(vfs_handle_struct *handle,
    2446             :                                        files_struct *fsp, void *data,
    2447             :                                        size_t n, off_t offset)
    2448             : {
    2449          20 :         return SMB_VFS_NEXT_PREAD(handle, fsp, data, n, offset);
    2450             : }
    2451             : 
    2452           0 : static ssize_t fruit_pread_rsrc_xattr(vfs_handle_struct *handle,
    2453             :                                       files_struct *fsp, void *data,
    2454             :                                       size_t n, off_t offset)
    2455             : {
    2456           0 :         return SMB_VFS_NEXT_PREAD(handle, fsp, data, n, offset);
    2457             : }
    2458             : 
    2459          76 : static ssize_t fruit_pread_rsrc_adouble(vfs_handle_struct *handle,
    2460             :                                         files_struct *fsp, void *data,
    2461             :                                         size_t n, off_t offset)
    2462             : {
    2463          76 :         struct fio *fio = fruit_get_complete_fio(handle, fsp);
    2464          76 :         struct adouble *ad = NULL;
    2465             :         ssize_t nread;
    2466             : 
    2467          76 :         if (fio == NULL || fio->ad_fsp == NULL) {
    2468           0 :                 DBG_ERR("fio/ad_fsp=NULL for [%s]\n", fsp_str_dbg(fsp));
    2469           0 :                 errno = EBADF;
    2470           0 :                 return -1;
    2471             :         }
    2472             : 
    2473          76 :         ad = ad_fget(talloc_tos(), handle, fio->ad_fsp, ADOUBLE_RSRC);
    2474          76 :         if (ad == NULL) {
    2475           0 :                 DBG_ERR("ad_fget [%s] failed [%s]\n",
    2476             :                         fsp_str_dbg(fio->ad_fsp), strerror(errno));
    2477           0 :                 return -1;
    2478             :         }
    2479             : 
    2480          76 :         nread = SMB_VFS_NEXT_PREAD(handle, fio->ad_fsp, data, n,
    2481             :                                    offset + ad_getentryoff(ad, ADEID_RFORK));
    2482             : 
    2483          76 :         TALLOC_FREE(ad);
    2484          76 :         return nread;
    2485             : }
    2486             : 
    2487          96 : static ssize_t fruit_pread_rsrc(vfs_handle_struct *handle,
    2488             :                                 files_struct *fsp, void *data,
    2489             :                                 size_t n, off_t offset)
    2490             : {
    2491          96 :         struct fio *fio = fruit_get_complete_fio(handle, fsp);
    2492             :         ssize_t nread;
    2493             : 
    2494          96 :         if (fio == NULL) {
    2495           0 :                 errno = EINVAL;
    2496           0 :                 return -1;
    2497             :         }
    2498             : 
    2499          96 :         switch (fio->config->rsrc) {
    2500          20 :         case FRUIT_RSRC_STREAM:
    2501          20 :                 nread = fruit_pread_rsrc_stream(handle, fsp, data, n, offset);
    2502          20 :                 break;
    2503             : 
    2504          76 :         case FRUIT_RSRC_ADFILE:
    2505          76 :                 nread = fruit_pread_rsrc_adouble(handle, fsp, data, n, offset);
    2506          76 :                 break;
    2507             : 
    2508           0 :         case FRUIT_RSRC_XATTR:
    2509           0 :                 nread = fruit_pread_rsrc_xattr(handle, fsp, data, n, offset);
    2510           0 :                 break;
    2511             : 
    2512           0 :         default:
    2513           0 :                 DBG_ERR("Unexpected rsrc config [%d]\n", fio->config->rsrc);
    2514           0 :                 return -1;
    2515             :         }
    2516             : 
    2517          96 :         return nread;
    2518             : }
    2519             : 
    2520         750 : static ssize_t fruit_pread(vfs_handle_struct *handle,
    2521             :                            files_struct *fsp, void *data,
    2522             :                            size_t n, off_t offset)
    2523             : {
    2524         750 :         struct fio *fio = fruit_get_complete_fio(handle, fsp);
    2525             :         ssize_t nread;
    2526             : 
    2527         750 :         DBG_DEBUG("Path [%s] offset=%"PRIdMAX", size=%zd\n",
    2528             :                   fsp_str_dbg(fsp), (intmax_t)offset, n);
    2529             : 
    2530         750 :         if (fio == NULL) {
    2531         102 :                 return SMB_VFS_NEXT_PREAD(handle, fsp, data, n, offset);
    2532             :         }
    2533             : 
    2534         648 :         if (fio->type == ADOUBLE_META) {
    2535         552 :                 nread = fruit_pread_meta(handle, fsp, data, n, offset);
    2536             :         } else {
    2537          96 :                 nread = fruit_pread_rsrc(handle, fsp, data, n, offset);
    2538             :         }
    2539             : 
    2540         648 :         DBG_DEBUG("Path [%s] nread [%zd]\n", fsp_str_dbg(fsp), nread);
    2541         648 :         return nread;
    2542             : }
    2543             : 
    2544         116 : static bool fruit_must_handle_aio_stream(struct fio *fio)
    2545             : {
    2546         116 :         if (fio == NULL) {
    2547          76 :                 return false;
    2548             :         };
    2549             : 
    2550          40 :         if (fio->type == ADOUBLE_META) {
    2551          16 :                 return true;
    2552             :         }
    2553             : 
    2554          24 :         if ((fio->type == ADOUBLE_RSRC) &&
    2555          24 :             (fio->config->rsrc == FRUIT_RSRC_ADFILE))
    2556             :         {
    2557          18 :                 return true;
    2558             :         }
    2559             : 
    2560           6 :         return false;
    2561             : }
    2562             : 
    2563             : struct fruit_pread_state {
    2564             :         ssize_t nread;
    2565             :         struct vfs_aio_state vfs_aio_state;
    2566             : };
    2567             : 
    2568             : static void fruit_pread_done(struct tevent_req *subreq);
    2569             : 
    2570          32 : static struct tevent_req *fruit_pread_send(
    2571             :         struct vfs_handle_struct *handle,
    2572             :         TALLOC_CTX *mem_ctx,
    2573             :         struct tevent_context *ev,
    2574             :         struct files_struct *fsp,
    2575             :         void *data,
    2576             :         size_t n, off_t offset)
    2577             : {
    2578          32 :         struct tevent_req *req = NULL;
    2579          32 :         struct tevent_req *subreq = NULL;
    2580          32 :         struct fruit_pread_state *state = NULL;
    2581          32 :         struct fio *fio = fruit_get_complete_fio(handle, fsp);
    2582             : 
    2583          32 :         req = tevent_req_create(mem_ctx, &state,
    2584             :                                 struct fruit_pread_state);
    2585          32 :         if (req == NULL) {
    2586           0 :                 return NULL;
    2587             :         }
    2588             : 
    2589          32 :         if (fruit_must_handle_aio_stream(fio)) {
    2590          14 :                 state->nread = SMB_VFS_PREAD(fsp, data, n, offset);
    2591          14 :                 if (state->nread != n) {
    2592           0 :                         if (state->nread != -1) {
    2593           0 :                                 errno = EIO;
    2594             :                         }
    2595           0 :                         tevent_req_error(req, errno);
    2596           0 :                         return tevent_req_post(req, ev);
    2597             :                 }
    2598          14 :                 tevent_req_done(req);
    2599          14 :                 return tevent_req_post(req, ev);
    2600             :         }
    2601             : 
    2602          18 :         subreq = SMB_VFS_NEXT_PREAD_SEND(state, ev, handle, fsp,
    2603             :                                          data, n, offset);
    2604          18 :         if (tevent_req_nomem(req, subreq)) {
    2605           0 :                 return tevent_req_post(req, ev);
    2606             :         }
    2607          18 :         tevent_req_set_callback(subreq, fruit_pread_done, req);
    2608          18 :         return req;
    2609             : }
    2610             : 
    2611          18 : static void fruit_pread_done(struct tevent_req *subreq)
    2612             : {
    2613          18 :         struct tevent_req *req = tevent_req_callback_data(
    2614             :                 subreq, struct tevent_req);
    2615          18 :         struct fruit_pread_state *state = tevent_req_data(
    2616             :                 req, struct fruit_pread_state);
    2617             : 
    2618          18 :         state->nread = SMB_VFS_PREAD_RECV(subreq, &state->vfs_aio_state);
    2619          18 :         TALLOC_FREE(subreq);
    2620             : 
    2621          18 :         if (tevent_req_error(req, state->vfs_aio_state.error)) {
    2622           0 :                 return;
    2623             :         }
    2624          18 :         tevent_req_done(req);
    2625             : }
    2626             : 
    2627          32 : static ssize_t fruit_pread_recv(struct tevent_req *req,
    2628             :                                         struct vfs_aio_state *vfs_aio_state)
    2629             : {
    2630          32 :         struct fruit_pread_state *state = tevent_req_data(
    2631             :                 req, struct fruit_pread_state);
    2632          32 :         ssize_t retval = -1;
    2633             : 
    2634          32 :         if (tevent_req_is_unix_error(req, &vfs_aio_state->error)) {
    2635           0 :                 tevent_req_received(req);
    2636           0 :                 return -1;
    2637             :         }
    2638             : 
    2639          32 :         *vfs_aio_state = state->vfs_aio_state;
    2640          32 :         retval = state->nread;
    2641          32 :         tevent_req_received(req);
    2642          32 :         return retval;
    2643             : }
    2644             : 
    2645         404 : static ssize_t fruit_pwrite_meta_stream(vfs_handle_struct *handle,
    2646             :                                         files_struct *fsp, const void *indata,
    2647             :                                         size_t n, off_t offset)
    2648             : {
    2649         404 :         struct fio *fio = fruit_get_complete_fio(handle, fsp);
    2650         404 :         const void *data = indata;
    2651             :         char afpinfo_buf[AFP_INFO_SIZE];
    2652         404 :         AfpInfo *ai = NULL;
    2653             :         size_t nwritten;
    2654             :         int ret;
    2655             :         bool ok;
    2656             : 
    2657         404 :         DBG_DEBUG("Path [%s] offset=%"PRIdMAX", size=%zd\n",
    2658             :                   fsp_str_dbg(fsp), (intmax_t)offset, n);
    2659             : 
    2660         404 :         if (fio == NULL) {
    2661           0 :                 return -1;
    2662             :         }
    2663             : 
    2664         404 :         if (fio->fake_fd) {
    2665         364 :                 struct vfs_open_how how = {
    2666         364 :                         .flags = fio->flags, .mode = fio->mode,
    2667             :                 };
    2668         364 :                 int fd = fsp_get_pathref_fd(fsp);
    2669             : 
    2670         364 :                 ret = vfs_fake_fd_close(fd);
    2671         364 :                 fsp_set_fd(fsp, -1);
    2672         364 :                 if (ret != 0) {
    2673           0 :                         DBG_ERR("Close [%s] failed: %s\n",
    2674             :                                 fsp_str_dbg(fsp), strerror(errno));
    2675           0 :                         return -1;
    2676             :                 }
    2677             : 
    2678         364 :                 fd = SMB_VFS_NEXT_OPENAT(handle,
    2679             :                                          NULL, /* opening a stream */
    2680             :                                          fsp->fsp_name,
    2681             :                                          fsp,
    2682             :                                          &how);
    2683         364 :                 if (fd == -1) {
    2684           0 :                         DBG_ERR("On-demand create [%s] in write failed: %s\n",
    2685             :                                 fsp_str_dbg(fsp), strerror(errno));
    2686           0 :                         return -1;
    2687             :                 }
    2688         364 :                 fsp_set_fd(fsp, fd);
    2689         364 :                 fio->fake_fd = false;
    2690             :         }
    2691             : 
    2692         404 :         ai = afpinfo_unpack(talloc_tos(), data, fio->config->validate_afpinfo);
    2693         404 :         if (ai == NULL) {
    2694           0 :                 return -1;
    2695             :         }
    2696             : 
    2697         404 :         if (ai_empty_finderinfo(ai)) {
    2698             :                 /*
    2699             :                  * Writing an all 0 blob to the metadata stream results in the
    2700             :                  * stream being removed on a macOS server. This ensures we
    2701             :                  * behave the same and it verified by the "delete AFP_AfpInfo by
    2702             :                  * writing all 0" test.
    2703             :                  */
    2704           6 :                 ret = SMB_VFS_NEXT_FTRUNCATE(handle, fsp, 0);
    2705           6 :                 if (ret != 0) {
    2706           0 :                         DBG_ERR("SMB_VFS_NEXT_FTRUNCATE on [%s] failed\n",
    2707             :                                 fsp_str_dbg(fsp));
    2708           0 :                         return -1;
    2709             :                 }
    2710             : 
    2711           6 :                 ok = set_delete_on_close(
    2712             :                         fsp,
    2713             :                         true,
    2714           6 :                         handle->conn->session_info->security_token,
    2715           6 :                         handle->conn->session_info->unix_token);
    2716           6 :                 if (!ok) {
    2717           0 :                         DBG_ERR("set_delete_on_close on [%s] failed\n",
    2718             :                                 fsp_str_dbg(fsp));
    2719           0 :                         return -1;
    2720             :                 }
    2721           6 :                 return n;
    2722             :         }
    2723             : 
    2724         398 :         if (!fio->config->validate_afpinfo) {
    2725             :                 /*
    2726             :                  * Ensure the buffer contains a valid header, so marshall
    2727             :                  * the data from the afpinfo struck back into a buffer
    2728             :                  * and write that instead of the possibly malformed data
    2729             :                  * we got from the client.
    2730             :                  */
    2731           4 :                 nwritten = afpinfo_pack(ai, afpinfo_buf);
    2732           4 :                 if (nwritten != AFP_INFO_SIZE) {
    2733           0 :                         errno = EINVAL;
    2734           0 :                         return -1;
    2735             :                 }
    2736           4 :                 data = afpinfo_buf;
    2737             :         }
    2738             : 
    2739         398 :         nwritten = SMB_VFS_NEXT_PWRITE(handle, fsp, data, n, offset);
    2740         398 :         if (nwritten != n) {
    2741           0 :                 return -1;
    2742             :         }
    2743             : 
    2744         398 :         return n;
    2745             : }
    2746             : 
    2747         136 : static ssize_t fruit_pwrite_meta_netatalk(vfs_handle_struct *handle,
    2748             :                                           files_struct *fsp, const void *data,
    2749             :                                           size_t n, off_t offset)
    2750             : {
    2751         136 :         struct fruit_config_data *config = NULL;
    2752         136 :         struct adouble *ad = NULL;
    2753         136 :         AfpInfo *ai = NULL;
    2754         136 :         char *p = NULL;
    2755             :         int ret;
    2756             :         bool ok;
    2757             : 
    2758         136 :         SMB_VFS_HANDLE_GET_DATA(handle, config,
    2759             :                                 struct fruit_config_data, return -1);
    2760             : 
    2761         136 :         ai = afpinfo_unpack(talloc_tos(), data, config->validate_afpinfo);
    2762         136 :         if (ai == NULL) {
    2763           0 :                 return -1;
    2764             :         }
    2765             : 
    2766         136 :         ad = ad_fget(talloc_tos(), handle, fsp, ADOUBLE_META);
    2767         136 :         if (ad == NULL) {
    2768         122 :                 ad = ad_init(talloc_tos(), ADOUBLE_META);
    2769         122 :                 if (ad == NULL) {
    2770           0 :                         return -1;
    2771             :                 }
    2772             :         }
    2773         136 :         p = ad_get_entry(ad, ADEID_FINDERI);
    2774         136 :         if (p == NULL) {
    2775           0 :                 DBG_ERR("No ADEID_FINDERI for [%s]\n", fsp_str_dbg(fsp));
    2776           0 :                 TALLOC_FREE(ad);
    2777           0 :                 return -1;
    2778             :         }
    2779             : 
    2780         136 :         memcpy(p, &ai->afpi_FinderInfo[0], ADEDLEN_FINDERI);
    2781             : 
    2782         136 :         ret = ad_fset(handle, ad, fsp);
    2783         136 :         if (ret != 0) {
    2784           0 :                 DBG_ERR("ad_pwrite [%s] failed\n", fsp_str_dbg(fsp));
    2785           0 :                 TALLOC_FREE(ad);
    2786           0 :                 return -1;
    2787             :         }
    2788             : 
    2789         136 :         TALLOC_FREE(ad);
    2790             : 
    2791         136 :         if (!ai_empty_finderinfo(ai)) {
    2792         134 :                 return n;
    2793             :         }
    2794             : 
    2795             :         /*
    2796             :          * Writing an all 0 blob to the metadata stream results in the stream
    2797             :          * being removed on a macOS server. This ensures we behave the same and
    2798             :          * it verified by the "delete AFP_AfpInfo by writing all 0" test.
    2799             :          */
    2800             : 
    2801           2 :         ok = set_delete_on_close(
    2802             :                 fsp,
    2803             :                 true,
    2804           2 :                 handle->conn->session_info->security_token,
    2805           2 :                 handle->conn->session_info->unix_token);
    2806           2 :         if (!ok) {
    2807           0 :                 DBG_ERR("set_delete_on_close on [%s] failed\n",
    2808             :                         fsp_str_dbg(fsp));
    2809           0 :                 return -1;
    2810             :         }
    2811             : 
    2812           2 :         return n;
    2813             : }
    2814             : 
    2815        2782 : static ssize_t fruit_pwrite_meta(vfs_handle_struct *handle,
    2816             :                                  files_struct *fsp, const void *data,
    2817             :                                  size_t n, off_t offset)
    2818             : {
    2819        2782 :         struct fio *fio = fruit_get_complete_fio(handle, fsp);
    2820             :         ssize_t nwritten;
    2821             :         uint8_t buf[AFP_INFO_SIZE];
    2822             :         size_t to_write;
    2823             :         size_t to_copy;
    2824             :         int cmp;
    2825             : 
    2826        2782 :         if (fio == NULL) {
    2827           0 :                 DBG_ERR("Failed to fetch fsp extension\n");
    2828           0 :                 return -1;
    2829             :         }
    2830             : 
    2831        2782 :         if (n < 3) {
    2832         256 :                 errno = EINVAL;
    2833         256 :                 return -1;
    2834             :         }
    2835             : 
    2836        2526 :         if (offset != 0 && n < 60) {
    2837        1568 :                 errno = EINVAL;
    2838        1568 :                 return -1;
    2839             :         }
    2840             : 
    2841         958 :         if (fio->config->validate_afpinfo) {
    2842         954 :                 cmp = memcmp(data, "AFP", 3);
    2843         954 :                 if (cmp != 0) {
    2844         378 :                         errno = EINVAL;
    2845         378 :                         return -1;
    2846             :                 }
    2847             :         }
    2848             : 
    2849         580 :         if (n <= AFP_OFF_FinderInfo) {
    2850             :                 /*
    2851             :                  * Nothing to do here really, just return
    2852             :                  */
    2853          40 :                 return n;
    2854             :         }
    2855             : 
    2856         540 :         offset = 0;
    2857             : 
    2858         540 :         to_copy = n;
    2859         540 :         if (to_copy > AFP_INFO_SIZE) {
    2860         200 :                 to_copy = AFP_INFO_SIZE;
    2861             :         }
    2862         540 :         memcpy(buf, data, to_copy);
    2863             : 
    2864         540 :         to_write = n;
    2865         540 :         if (to_write != AFP_INFO_SIZE) {
    2866         272 :                 to_write = AFP_INFO_SIZE;
    2867             :         }
    2868             : 
    2869         540 :         switch (fio->config->meta) {
    2870         404 :         case FRUIT_META_STREAM:
    2871         404 :                 nwritten = fruit_pwrite_meta_stream(handle,
    2872             :                                                     fsp,
    2873             :                                                     buf,
    2874             :                                                     to_write,
    2875             :                                                     offset);
    2876         404 :                 break;
    2877             : 
    2878         136 :         case FRUIT_META_NETATALK:
    2879         136 :                 nwritten = fruit_pwrite_meta_netatalk(handle,
    2880             :                                                       fsp,
    2881             :                                                       buf,
    2882             :                                                       to_write,
    2883             :                                                       offset);
    2884         136 :                 break;
    2885             : 
    2886           0 :         default:
    2887           0 :                 DBG_ERR("Unexpected meta config [%d]\n", fio->config->meta);
    2888           0 :                 return -1;
    2889             :         }
    2890             : 
    2891         540 :         if (nwritten != to_write) {
    2892           0 :                 return -1;
    2893             :         }
    2894             : 
    2895             :         /*
    2896             :          * Return the requested amount, verified against macOS SMB server
    2897             :          */
    2898         540 :         return n;
    2899             : }
    2900             : 
    2901          40 : static ssize_t fruit_pwrite_rsrc_stream(vfs_handle_struct *handle,
    2902             :                                         files_struct *fsp, const void *data,
    2903             :                                         size_t n, off_t offset)
    2904             : {
    2905          40 :         return SMB_VFS_NEXT_PWRITE(handle, fsp, data, n, offset);
    2906             : }
    2907             : 
    2908           0 : static ssize_t fruit_pwrite_rsrc_xattr(vfs_handle_struct *handle,
    2909             :                                        files_struct *fsp, const void *data,
    2910             :                                        size_t n, off_t offset)
    2911             : {
    2912           0 :         return SMB_VFS_NEXT_PWRITE(handle, fsp, data, n, offset);
    2913             : }
    2914             : 
    2915         124 : static ssize_t fruit_pwrite_rsrc_adouble(vfs_handle_struct *handle,
    2916             :                                          files_struct *fsp, const void *data,
    2917             :                                          size_t n, off_t offset)
    2918             : {
    2919         124 :         struct fio *fio = fruit_get_complete_fio(handle, fsp);
    2920         124 :         struct adouble *ad = NULL;
    2921             :         ssize_t nwritten;
    2922             :         int ret;
    2923             : 
    2924         124 :         if (fio == NULL || fio->ad_fsp == NULL) {
    2925           0 :                 DBG_ERR("fio/ad_fsp=NULL for [%s]\n", fsp_str_dbg(fsp));
    2926           0 :                 errno = EBADF;
    2927           0 :                 return -1;
    2928             :         }
    2929             : 
    2930         124 :         ad = ad_fget(talloc_tos(), handle, fio->ad_fsp, ADOUBLE_RSRC);
    2931         124 :         if (ad == NULL) {
    2932           0 :                 DBG_ERR("ad_fget [%s] failed [%s]\n",
    2933             :                         fsp_str_dbg(fio->ad_fsp), strerror(errno));
    2934           0 :                 return -1;
    2935             :         }
    2936             : 
    2937         124 :         nwritten = SMB_VFS_NEXT_PWRITE(handle, fio->ad_fsp, data, n,
    2938             :                                        offset + ad_getentryoff(ad, ADEID_RFORK));
    2939         124 :         if (nwritten != n) {
    2940           0 :                 DBG_ERR("Short write on [%s] [%zd/%zd]\n",
    2941             :                         fsp_str_dbg(fio->ad_fsp), nwritten, n);
    2942           0 :                 TALLOC_FREE(ad);
    2943           0 :                 return -1;
    2944             :         }
    2945             : 
    2946         124 :         if ((n + offset) > ad_getentrylen(ad, ADEID_RFORK)) {
    2947         118 :                 ad_setentrylen(ad, ADEID_RFORK, n + offset);
    2948         118 :                 ret = ad_fset(handle, ad, fio->ad_fsp);
    2949         118 :                 if (ret != 0) {
    2950           0 :                         DBG_ERR("ad_pwrite [%s] failed\n", fsp_str_dbg(fio->ad_fsp));
    2951           0 :                         TALLOC_FREE(ad);
    2952           0 :                         return -1;
    2953             :                 }
    2954             :         }
    2955             : 
    2956         124 :         TALLOC_FREE(ad);
    2957         124 :         return n;
    2958             : }
    2959             : 
    2960         164 : static ssize_t fruit_pwrite_rsrc(vfs_handle_struct *handle,
    2961             :                                  files_struct *fsp, const void *data,
    2962             :                                  size_t n, off_t offset)
    2963             : {
    2964         164 :         struct fio *fio = fruit_get_complete_fio(handle, fsp);
    2965             :         ssize_t nwritten;
    2966             : 
    2967         164 :         if (fio == NULL) {
    2968           0 :                 DBG_ERR("Failed to fetch fsp extension\n");
    2969           0 :                 return -1;
    2970             :         }
    2971             : 
    2972         164 :         switch (fio->config->rsrc) {
    2973          40 :         case FRUIT_RSRC_STREAM:
    2974          40 :                 nwritten = fruit_pwrite_rsrc_stream(handle, fsp, data, n, offset);
    2975          40 :                 break;
    2976             : 
    2977         124 :         case FRUIT_RSRC_ADFILE:
    2978         124 :                 nwritten = fruit_pwrite_rsrc_adouble(handle, fsp, data, n, offset);
    2979         124 :                 break;
    2980             : 
    2981           0 :         case FRUIT_RSRC_XATTR:
    2982           0 :                 nwritten = fruit_pwrite_rsrc_xattr(handle, fsp, data, n, offset);
    2983           0 :                 break;
    2984             : 
    2985           0 :         default:
    2986           0 :                 DBG_ERR("Unexpected rsrc config [%d]\n", fio->config->rsrc);
    2987           0 :                 return -1;
    2988             :         }
    2989             : 
    2990         164 :         return nwritten;
    2991             : }
    2992             : 
    2993        3170 : static ssize_t fruit_pwrite(vfs_handle_struct *handle,
    2994             :                             files_struct *fsp, const void *data,
    2995             :                             size_t n, off_t offset)
    2996             : {
    2997        3170 :         struct fio *fio = fruit_get_complete_fio(handle, fsp);
    2998             :         ssize_t nwritten;
    2999             : 
    3000        3170 :         DBG_DEBUG("Path [%s] offset=%"PRIdMAX", size=%zd\n",
    3001             :                   fsp_str_dbg(fsp), (intmax_t)offset, n);
    3002             : 
    3003        3170 :         if (fio == NULL) {
    3004         224 :                 return SMB_VFS_NEXT_PWRITE(handle, fsp, data, n, offset);
    3005             :         }
    3006             : 
    3007        2946 :         if (fio->type == ADOUBLE_META) {
    3008        2782 :                 nwritten = fruit_pwrite_meta(handle, fsp, data, n, offset);
    3009             :         } else {
    3010         164 :                 nwritten = fruit_pwrite_rsrc(handle, fsp, data, n, offset);
    3011             :         }
    3012             : 
    3013        2946 :         DBG_DEBUG("Path [%s] nwritten=%zd\n", fsp_str_dbg(fsp), nwritten);
    3014        2946 :         return nwritten;
    3015             : }
    3016             : 
    3017             : struct fruit_pwrite_state {
    3018             :         ssize_t nwritten;
    3019             :         struct vfs_aio_state vfs_aio_state;
    3020             : };
    3021             : 
    3022             : static void fruit_pwrite_done(struct tevent_req *subreq);
    3023             : 
    3024          76 : static struct tevent_req *fruit_pwrite_send(
    3025             :         struct vfs_handle_struct *handle,
    3026             :         TALLOC_CTX *mem_ctx,
    3027             :         struct tevent_context *ev,
    3028             :         struct files_struct *fsp,
    3029             :         const void *data,
    3030             :         size_t n, off_t offset)
    3031             : {
    3032          76 :         struct tevent_req *req = NULL;
    3033          76 :         struct tevent_req *subreq = NULL;
    3034          76 :         struct fruit_pwrite_state *state = NULL;
    3035          76 :         struct fio *fio = fruit_get_complete_fio(handle, fsp);
    3036             : 
    3037          76 :         req = tevent_req_create(mem_ctx, &state,
    3038             :                                 struct fruit_pwrite_state);
    3039          76 :         if (req == NULL) {
    3040           0 :                 return NULL;
    3041             :         }
    3042             : 
    3043          76 :         if (fruit_must_handle_aio_stream(fio)) {
    3044          14 :                 state->nwritten = SMB_VFS_PWRITE(fsp, data, n, offset);
    3045          14 :                 if (state->nwritten != n) {
    3046           0 :                         if (state->nwritten != -1) {
    3047           0 :                                 errno = EIO;
    3048             :                         }
    3049           0 :                         tevent_req_error(req, errno);
    3050           0 :                         return tevent_req_post(req, ev);
    3051             :                 }
    3052          14 :                 tevent_req_done(req);
    3053          14 :                 return tevent_req_post(req, ev);
    3054             :         }
    3055             : 
    3056          62 :         subreq = SMB_VFS_NEXT_PWRITE_SEND(state, ev, handle, fsp,
    3057             :                                           data, n, offset);
    3058          62 :         if (tevent_req_nomem(req, subreq)) {
    3059           0 :                 return tevent_req_post(req, ev);
    3060             :         }
    3061          62 :         tevent_req_set_callback(subreq, fruit_pwrite_done, req);
    3062          62 :         return req;
    3063             : }
    3064             : 
    3065          62 : static void fruit_pwrite_done(struct tevent_req *subreq)
    3066             : {
    3067          62 :         struct tevent_req *req = tevent_req_callback_data(
    3068             :                 subreq, struct tevent_req);
    3069          62 :         struct fruit_pwrite_state *state = tevent_req_data(
    3070             :                 req, struct fruit_pwrite_state);
    3071             : 
    3072          62 :         state->nwritten = SMB_VFS_PWRITE_RECV(subreq, &state->vfs_aio_state);
    3073          62 :         TALLOC_FREE(subreq);
    3074             : 
    3075          62 :         if (tevent_req_error(req, state->vfs_aio_state.error)) {
    3076           0 :                 return;
    3077             :         }
    3078          62 :         tevent_req_done(req);
    3079             : }
    3080             : 
    3081          76 : static ssize_t fruit_pwrite_recv(struct tevent_req *req,
    3082             :                                          struct vfs_aio_state *vfs_aio_state)
    3083             : {
    3084          76 :         struct fruit_pwrite_state *state = tevent_req_data(
    3085             :                 req, struct fruit_pwrite_state);
    3086          76 :         ssize_t retval = -1;
    3087             : 
    3088          76 :         if (tevent_req_is_unix_error(req, &vfs_aio_state->error)) {
    3089           0 :                 tevent_req_received(req);
    3090           0 :                 return -1;
    3091             :         }
    3092             : 
    3093          76 :         *vfs_aio_state = state->vfs_aio_state;
    3094          76 :         retval = state->nwritten;
    3095          76 :         tevent_req_received(req);
    3096          76 :         return retval;
    3097             : }
    3098             : 
    3099             : struct fruit_fsync_state {
    3100             :         int ret;
    3101             :         struct vfs_aio_state vfs_aio_state;
    3102             : };
    3103             : 
    3104             : static void fruit_fsync_done(struct tevent_req *subreq);
    3105             : 
    3106           8 : static struct tevent_req *fruit_fsync_send(
    3107             :         struct vfs_handle_struct *handle,
    3108             :         TALLOC_CTX *mem_ctx,
    3109             :         struct tevent_context *ev,
    3110             :         struct files_struct *fsp)
    3111             : {
    3112           8 :         struct tevent_req *req = NULL;
    3113           8 :         struct tevent_req *subreq = NULL;
    3114           8 :         struct fruit_fsync_state *state = NULL;
    3115           8 :         struct fio *fio = fruit_get_complete_fio(handle, fsp);
    3116             : 
    3117           8 :         req = tevent_req_create(mem_ctx, &state,
    3118             :                                 struct fruit_fsync_state);
    3119           8 :         if (req == NULL) {
    3120           0 :                 return NULL;
    3121             :         }
    3122             : 
    3123           8 :         if (fruit_must_handle_aio_stream(fio)) {
    3124           6 :                 struct adouble *ad = NULL;
    3125             : 
    3126           6 :                 if (fio->type == ADOUBLE_META) {
    3127             :                         /*
    3128             :                          * We must never pass a fake_fd
    3129             :                          * to lower level fsync calls.
    3130             :                          * Everything is already done
    3131             :                          * synchronously, so just return
    3132             :                          * true.
    3133             :                          */
    3134           0 :                         SMB_ASSERT(fio->fake_fd);
    3135           0 :                         tevent_req_done(req);
    3136           0 :                         return tevent_req_post(req, ev);
    3137             :                 }
    3138             : 
    3139             :                 /*
    3140             :                  * We know the following must be true,
    3141             :                  * as it's the condition for fruit_must_handle_aio_stream()
    3142             :                  * to return true if fio->type == ADOUBLE_RSRC.
    3143             :                  */
    3144           6 :                 SMB_ASSERT(fio->config->rsrc == FRUIT_RSRC_ADFILE);
    3145           6 :                 if (fio->ad_fsp == NULL) {
    3146           0 :                         tevent_req_error(req, EBADF);
    3147           0 :                         return tevent_req_post(req, ev);
    3148             :                 }
    3149           6 :                 ad = ad_fget(talloc_tos(), handle, fio->ad_fsp, ADOUBLE_RSRC);
    3150           6 :                 if (ad == NULL) {
    3151           0 :                         tevent_req_error(req, ENOMEM);
    3152           0 :                         return tevent_req_post(req, ev);
    3153             :                 }
    3154           6 :                 fsp = fio->ad_fsp;
    3155             :         }
    3156             : 
    3157           8 :         subreq = SMB_VFS_NEXT_FSYNC_SEND(state, ev, handle, fsp);
    3158           8 :         if (tevent_req_nomem(req, subreq)) {
    3159           0 :                 return tevent_req_post(req, ev);
    3160             :         }
    3161           8 :         tevent_req_set_callback(subreq, fruit_fsync_done, req);
    3162           8 :         return req;
    3163             : }
    3164             : 
    3165           8 : static void fruit_fsync_done(struct tevent_req *subreq)
    3166             : {
    3167           8 :         struct tevent_req *req = tevent_req_callback_data(
    3168             :                 subreq, struct tevent_req);
    3169           8 :         struct fruit_fsync_state *state = tevent_req_data(
    3170             :                 req, struct fruit_fsync_state);
    3171             : 
    3172           8 :         state->ret = SMB_VFS_FSYNC_RECV(subreq, &state->vfs_aio_state);
    3173           8 :         TALLOC_FREE(subreq);
    3174           8 :         if (state->ret != 0) {
    3175           0 :                 tevent_req_error(req, errno);
    3176           0 :                 return;
    3177             :         }
    3178           8 :         tevent_req_done(req);
    3179             : }
    3180             : 
    3181           8 : static int fruit_fsync_recv(struct tevent_req *req,
    3182             :                                         struct vfs_aio_state *vfs_aio_state)
    3183             : {
    3184           8 :         struct fruit_fsync_state *state = tevent_req_data(
    3185             :                 req, struct fruit_fsync_state);
    3186           8 :         int retval = -1;
    3187             : 
    3188           8 :         if (tevent_req_is_unix_error(req, &vfs_aio_state->error)) {
    3189           0 :                 tevent_req_received(req);
    3190           0 :                 return -1;
    3191             :         }
    3192             : 
    3193           8 :         *vfs_aio_state = state->vfs_aio_state;
    3194           8 :         retval = state->ret;
    3195           8 :         tevent_req_received(req);
    3196           8 :         return retval;
    3197             : }
    3198             : 
    3199             : /**
    3200             :  * Helper to stat/lstat the base file of an smb_fname.
    3201             :  */
    3202        5016 : static int fruit_stat_base(vfs_handle_struct *handle,
    3203             :                            struct smb_filename *smb_fname,
    3204             :                            bool follow_links)
    3205             : {
    3206             :         char *tmp_stream_name;
    3207             :         int rc;
    3208             : 
    3209        5016 :         tmp_stream_name = smb_fname->stream_name;
    3210        5016 :         smb_fname->stream_name = NULL;
    3211        5016 :         if (follow_links) {
    3212           6 :                 rc = SMB_VFS_NEXT_STAT(handle, smb_fname);
    3213             :         } else {
    3214        5010 :                 rc = SMB_VFS_NEXT_LSTAT(handle, smb_fname);
    3215             :         }
    3216        5016 :         smb_fname->stream_name = tmp_stream_name;
    3217             : 
    3218        5016 :         DBG_DEBUG("fruit_stat_base [%s] dev [%ju] ino [%ju]\n",
    3219             :                   smb_fname->base_name,
    3220             :                   (uintmax_t)smb_fname->st.st_ex_dev,
    3221             :                   (uintmax_t)smb_fname->st.st_ex_ino);
    3222        5016 :         return rc;
    3223             : }
    3224             : 
    3225           0 : static int fruit_stat_meta_stream(vfs_handle_struct *handle,
    3226             :                                   struct smb_filename *smb_fname,
    3227             :                                   bool follow_links)
    3228             : {
    3229             :         int ret;
    3230             :         ino_t ino;
    3231             : 
    3232           0 :         ret = fruit_stat_base(handle, smb_fname, false);
    3233           0 :         if (ret != 0) {
    3234           0 :                 return -1;
    3235             :         }
    3236             : 
    3237           0 :         ino = hash_inode(&smb_fname->st, smb_fname->stream_name);
    3238             : 
    3239           0 :         if (follow_links) {
    3240           0 :                 ret = SMB_VFS_NEXT_STAT(handle, smb_fname);
    3241             :         } else {
    3242           0 :                 ret = SMB_VFS_NEXT_LSTAT(handle, smb_fname);
    3243             :         }
    3244             : 
    3245           0 :         smb_fname->st.st_ex_ino = ino;
    3246             : 
    3247           0 :         return ret;
    3248             : }
    3249             : 
    3250           0 : static int fruit_stat_meta_netatalk(vfs_handle_struct *handle,
    3251             :                                     struct smb_filename *smb_fname,
    3252             :                                     bool follow_links)
    3253             : {
    3254           0 :         struct adouble *ad = NULL;
    3255             : 
    3256             :         /* Populate the stat struct with info from the base file. */
    3257           0 :         if (fruit_stat_base(handle, smb_fname, follow_links) == -1) {
    3258           0 :                 return -1;
    3259             :         }
    3260             : 
    3261           0 :         ad = ad_get_meta_fsp(talloc_tos(), handle, smb_fname);
    3262           0 :         if (ad == NULL) {
    3263           0 :                 DBG_INFO("fruit_stat_meta %s: %s\n",
    3264             :                          smb_fname_str_dbg(smb_fname), strerror(errno));
    3265           0 :                 errno = ENOENT;
    3266           0 :                 return -1;
    3267             :         }
    3268           0 :         TALLOC_FREE(ad);
    3269             : 
    3270           0 :         smb_fname->st.st_ex_size = AFP_INFO_SIZE;
    3271           0 :         smb_fname->st.st_ex_ino = hash_inode(&smb_fname->st,
    3272           0 :                                               smb_fname->stream_name);
    3273           0 :         return 0;
    3274             : }
    3275             : 
    3276           0 : static int fruit_stat_meta(vfs_handle_struct *handle,
    3277             :                            struct smb_filename *smb_fname,
    3278             :                            bool follow_links)
    3279             : {
    3280           0 :         struct fruit_config_data *config = NULL;
    3281             :         int ret;
    3282             : 
    3283           0 :         SMB_VFS_HANDLE_GET_DATA(handle, config,
    3284             :                                 struct fruit_config_data, return -1);
    3285             : 
    3286           0 :         switch (config->meta) {
    3287           0 :         case FRUIT_META_STREAM:
    3288           0 :                 ret = fruit_stat_meta_stream(handle, smb_fname, follow_links);
    3289           0 :                 break;
    3290             : 
    3291           0 :         case FRUIT_META_NETATALK:
    3292           0 :                 ret = fruit_stat_meta_netatalk(handle, smb_fname, follow_links);
    3293           0 :                 break;
    3294             : 
    3295           0 :         default:
    3296           0 :                 DBG_ERR("Unexpected meta config [%d]\n", config->meta);
    3297           0 :                 return -1;
    3298             :         }
    3299             : 
    3300           0 :         return ret;
    3301             : }
    3302             : 
    3303          12 : static int fruit_stat_rsrc_netatalk(vfs_handle_struct *handle,
    3304             :                                     struct smb_filename *smb_fname,
    3305             :                                     bool follow_links)
    3306             : {
    3307          12 :         struct adouble *ad = NULL;
    3308             :         int ret;
    3309             : 
    3310          12 :         ad = ad_get(talloc_tos(), handle, smb_fname, ADOUBLE_RSRC);
    3311          12 :         if (ad == NULL) {
    3312           6 :                 errno = ENOENT;
    3313           6 :                 return -1;
    3314             :         }
    3315             : 
    3316             :         /* Populate the stat struct with info from the base file. */
    3317           6 :         ret = fruit_stat_base(handle, smb_fname, follow_links);
    3318           6 :         if (ret != 0) {
    3319           0 :                 TALLOC_FREE(ad);
    3320           0 :                 return -1;
    3321             :         }
    3322             : 
    3323           6 :         smb_fname->st.st_ex_size = ad_getentrylen(ad, ADEID_RFORK);
    3324          12 :         smb_fname->st.st_ex_ino = hash_inode(&smb_fname->st,
    3325           6 :                                               smb_fname->stream_name);
    3326           6 :         TALLOC_FREE(ad);
    3327           6 :         return 0;
    3328             : }
    3329             : 
    3330          68 : static int fruit_stat_rsrc_stream(vfs_handle_struct *handle,
    3331             :                                   struct smb_filename *smb_fname,
    3332             :                                   bool follow_links)
    3333             : {
    3334             :         int ret;
    3335             : 
    3336          68 :         if (follow_links) {
    3337          68 :                 ret = SMB_VFS_NEXT_STAT(handle, smb_fname);
    3338             :         } else {
    3339           0 :                 ret = SMB_VFS_NEXT_LSTAT(handle, smb_fname);
    3340             :         }
    3341             : 
    3342          68 :         return ret;
    3343             : }
    3344             : 
    3345           0 : static int fruit_stat_rsrc_xattr(vfs_handle_struct *handle,
    3346             :                                  struct smb_filename *smb_fname,
    3347             :                                  bool follow_links)
    3348             : {
    3349             : #ifdef HAVE_ATTROPEN
    3350             :         int ret;
    3351             :         int fd = -1;
    3352             : 
    3353             :         /* Populate the stat struct with info from the base file. */
    3354             :         ret = fruit_stat_base(handle, smb_fname, follow_links);
    3355             :         if (ret != 0) {
    3356             :                 return -1;
    3357             :         }
    3358             : 
    3359             :         fd = attropen(smb_fname->base_name,
    3360             :                       AFPRESOURCE_EA_NETATALK,
    3361             :                       O_RDONLY);
    3362             :         if (fd == -1) {
    3363             :                 return 0;
    3364             :         }
    3365             : 
    3366             :         ret = sys_fstat(fd, &smb_fname->st, false);
    3367             :         if (ret != 0) {
    3368             :                 close(fd);
    3369             :                 DBG_ERR("fstat [%s:%s] failed\n", smb_fname->base_name,
    3370             :                         AFPRESOURCE_EA_NETATALK);
    3371             :                 return -1;
    3372             :         }
    3373             :         close(fd);
    3374             :         fd = -1;
    3375             : 
    3376             :         smb_fname->st.st_ex_ino = hash_inode(&smb_fname->st,
    3377             :                                              smb_fname->stream_name);
    3378             : 
    3379             :         return ret;
    3380             : 
    3381             : #else
    3382           0 :         errno = ENOSYS;
    3383           0 :         return -1;
    3384             : #endif
    3385             : }
    3386             : 
    3387          80 : static int fruit_stat_rsrc(vfs_handle_struct *handle,
    3388             :                            struct smb_filename *smb_fname,
    3389             :                            bool follow_links)
    3390             : {
    3391          80 :         struct fruit_config_data *config = NULL;
    3392             :         int ret;
    3393             : 
    3394          80 :         DBG_DEBUG("Path [%s]\n", smb_fname_str_dbg(smb_fname));
    3395             : 
    3396          80 :         SMB_VFS_HANDLE_GET_DATA(handle, config,
    3397             :                                 struct fruit_config_data, return -1);
    3398             : 
    3399          80 :         switch (config->rsrc) {
    3400          68 :         case FRUIT_RSRC_STREAM:
    3401          68 :                 ret = fruit_stat_rsrc_stream(handle, smb_fname, follow_links);
    3402          68 :                 break;
    3403             : 
    3404           0 :         case FRUIT_RSRC_XATTR:
    3405           0 :                 ret = fruit_stat_rsrc_xattr(handle, smb_fname, follow_links);
    3406           0 :                 break;
    3407             : 
    3408          12 :         case FRUIT_RSRC_ADFILE:
    3409          12 :                 ret = fruit_stat_rsrc_netatalk(handle, smb_fname, follow_links);
    3410          12 :                 break;
    3411             : 
    3412           0 :         default:
    3413           0 :                 DBG_ERR("Unexpected rsrc config [%d]\n", config->rsrc);
    3414           0 :                 return -1;
    3415             :         }
    3416             : 
    3417          80 :         return ret;
    3418             : }
    3419             : 
    3420      121602 : static int fruit_stat(vfs_handle_struct *handle,
    3421             :                       struct smb_filename *smb_fname)
    3422             : {
    3423      121602 :         int rc = -1;
    3424             : 
    3425      121602 :         DEBUG(10, ("fruit_stat called for %s\n",
    3426             :                    smb_fname_str_dbg(smb_fname)));
    3427             : 
    3428      121602 :         if (!is_named_stream(smb_fname)) {
    3429      121506 :                 rc = SMB_VFS_NEXT_STAT(handle, smb_fname);
    3430      121506 :                 if (rc == 0) {
    3431      121478 :                         update_btime(handle, smb_fname);
    3432             :                 }
    3433      121506 :                 return rc;
    3434             :         }
    3435             : 
    3436             :         /*
    3437             :          * Note if lp_posix_paths() is true, we can never
    3438             :          * get here as is_ntfs_stream_smb_fname() is
    3439             :          * always false. So we never need worry about
    3440             :          * not following links here.
    3441             :          */
    3442             : 
    3443          96 :         if (is_afpinfo_stream(smb_fname->stream_name)) {
    3444           0 :                 rc = fruit_stat_meta(handle, smb_fname, true);
    3445          96 :         } else if (is_afpresource_stream(smb_fname->stream_name)) {
    3446          80 :                 rc = fruit_stat_rsrc(handle, smb_fname, true);
    3447             :         } else {
    3448          16 :                 return SMB_VFS_NEXT_STAT(handle, smb_fname);
    3449             :         }
    3450             : 
    3451          80 :         if (rc == 0) {
    3452          16 :                 update_btime(handle, smb_fname);
    3453          16 :                 smb_fname->st.st_ex_mode &= ~S_IFMT;
    3454          16 :                 smb_fname->st.st_ex_mode |= S_IFREG;
    3455          16 :                 smb_fname->st.st_ex_blocks =
    3456          16 :                         smb_fname->st.st_ex_size / STAT_ST_BLOCKSIZE + 1;
    3457             :         }
    3458          80 :         return rc;
    3459             : }
    3460             : 
    3461         446 : static int fruit_lstat(vfs_handle_struct *handle,
    3462             :                        struct smb_filename *smb_fname)
    3463             : {
    3464         446 :         int rc = -1;
    3465             : 
    3466         446 :         DEBUG(10, ("fruit_lstat called for %s\n",
    3467             :                    smb_fname_str_dbg(smb_fname)));
    3468             : 
    3469         446 :         if (!is_named_stream(smb_fname)) {
    3470         446 :                 rc = SMB_VFS_NEXT_LSTAT(handle, smb_fname);
    3471         446 :                 if (rc == 0) {
    3472         446 :                         update_btime(handle, smb_fname);
    3473             :                 }
    3474         446 :                 return rc;
    3475             :         }
    3476             : 
    3477           0 :         if (is_afpinfo_stream(smb_fname->stream_name)) {
    3478           0 :                 rc = fruit_stat_meta(handle, smb_fname, false);
    3479           0 :         } else if (is_afpresource_stream(smb_fname->stream_name)) {
    3480           0 :                 rc = fruit_stat_rsrc(handle, smb_fname, false);
    3481             :         } else {
    3482           0 :                 return SMB_VFS_NEXT_LSTAT(handle, smb_fname);
    3483             :         }
    3484             : 
    3485           0 :         if (rc == 0) {
    3486           0 :                 update_btime(handle, smb_fname);
    3487           0 :                 smb_fname->st.st_ex_mode &= ~S_IFMT;
    3488           0 :                 smb_fname->st.st_ex_mode |= S_IFREG;
    3489           0 :                 smb_fname->st.st_ex_blocks =
    3490           0 :                         smb_fname->st.st_ex_size / STAT_ST_BLOCKSIZE + 1;
    3491             :         }
    3492           0 :         return rc;
    3493             : }
    3494             : 
    3495        3046 : static int fruit_fstat_meta_stream(vfs_handle_struct *handle,
    3496             :                                    files_struct *fsp,
    3497             :                                    SMB_STRUCT_STAT *sbuf)
    3498             : {
    3499        3046 :         struct fio *fio = fruit_get_complete_fio(handle, fsp);
    3500             :         struct smb_filename smb_fname;
    3501             :         ino_t ino;
    3502             :         int ret;
    3503             : 
    3504        3046 :         if (fio == NULL) {
    3505           0 :                 return -1;
    3506             :         }
    3507             : 
    3508        3046 :         if (fio->fake_fd) {
    3509        1240 :                 ret = fruit_stat_base(handle, fsp->base_fsp->fsp_name, false);
    3510        1240 :                 if (ret != 0) {
    3511           0 :                         return -1;
    3512             :                 }
    3513             : 
    3514        1240 :                 *sbuf = fsp->base_fsp->fsp_name->st;
    3515        1240 :                 sbuf->st_ex_size = AFP_INFO_SIZE;
    3516        1240 :                 sbuf->st_ex_ino = hash_inode(sbuf, fsp->fsp_name->stream_name);
    3517        1240 :                 return 0;
    3518             :         }
    3519             : 
    3520        1806 :         smb_fname = (struct smb_filename) {
    3521        1806 :                 .base_name = fsp->fsp_name->base_name,
    3522        1806 :                 .twrp = fsp->fsp_name->twrp,
    3523             :         };
    3524             : 
    3525        1806 :         ret = fruit_stat_base(handle, &smb_fname, false);
    3526        1806 :         if (ret != 0) {
    3527           0 :                 return -1;
    3528             :         }
    3529        1806 :         *sbuf = smb_fname.st;
    3530             : 
    3531        1806 :         ino = hash_inode(sbuf, fsp->fsp_name->stream_name);
    3532             : 
    3533        1806 :         ret = SMB_VFS_NEXT_FSTAT(handle, fsp, sbuf);
    3534        1806 :         if (ret != 0) {
    3535           0 :                 return -1;
    3536             :         }
    3537             : 
    3538        1806 :         sbuf->st_ex_ino = ino;
    3539        1806 :         return 0;
    3540             : }
    3541             : 
    3542         998 : static int fruit_fstat_meta_netatalk(vfs_handle_struct *handle,
    3543             :                                      files_struct *fsp,
    3544             :                                      SMB_STRUCT_STAT *sbuf)
    3545             : {
    3546             :         int ret;
    3547             : 
    3548         998 :         ret = fruit_stat_base(handle, fsp->base_fsp->fsp_name, false);
    3549         998 :         if (ret != 0) {
    3550           0 :                 return -1;
    3551             :         }
    3552             : 
    3553         998 :         *sbuf = fsp->base_fsp->fsp_name->st;
    3554         998 :         sbuf->st_ex_size = AFP_INFO_SIZE;
    3555         998 :         sbuf->st_ex_ino = hash_inode(sbuf, fsp->fsp_name->stream_name);
    3556             : 
    3557         998 :         return 0;
    3558             : }
    3559             : 
    3560        4044 : static int fruit_fstat_meta(vfs_handle_struct *handle,
    3561             :                             files_struct *fsp,
    3562             :                             SMB_STRUCT_STAT *sbuf,
    3563             :                             struct fio *fio)
    3564             : {
    3565             :         int ret;
    3566             : 
    3567        4044 :         DBG_DEBUG("Path [%s]\n", fsp_str_dbg(fsp));
    3568             : 
    3569        4044 :         switch (fio->config->meta) {
    3570        3046 :         case FRUIT_META_STREAM:
    3571        3046 :                 ret = fruit_fstat_meta_stream(handle, fsp, sbuf);
    3572        3046 :                 break;
    3573             : 
    3574         998 :         case FRUIT_META_NETATALK:
    3575         998 :                 ret = fruit_fstat_meta_netatalk(handle, fsp, sbuf);
    3576         998 :                 break;
    3577             : 
    3578           0 :         default:
    3579           0 :                 DBG_ERR("Unexpected meta config [%d]\n", fio->config->meta);
    3580           0 :                 return -1;
    3581             :         }
    3582             : 
    3583        4044 :         DBG_DEBUG("Path [%s] ret [%d]\n", fsp_str_dbg(fsp), ret);
    3584        4044 :         return ret;
    3585             : }
    3586             : 
    3587           0 : static int fruit_fstat_rsrc_xattr(vfs_handle_struct *handle,
    3588             :                                   files_struct *fsp,
    3589             :                                   SMB_STRUCT_STAT *sbuf)
    3590             : {
    3591           0 :         return SMB_VFS_NEXT_FSTAT(handle, fsp, sbuf);
    3592             : }
    3593             : 
    3594         384 : static int fruit_fstat_rsrc_stream(vfs_handle_struct *handle,
    3595             :                                    files_struct *fsp,
    3596             :                                    SMB_STRUCT_STAT *sbuf)
    3597             : {
    3598         384 :         return SMB_VFS_NEXT_FSTAT(handle, fsp, sbuf);
    3599             : }
    3600             : 
    3601         966 : static int fruit_fstat_rsrc_adouble(vfs_handle_struct *handle,
    3602             :                                     files_struct *fsp,
    3603             :                                     SMB_STRUCT_STAT *sbuf)
    3604             : {
    3605         966 :         struct fio *fio = fruit_get_complete_fio(handle, fsp);
    3606         966 :         struct adouble *ad = NULL;
    3607             :         int ret;
    3608             : 
    3609         966 :         if (fio == NULL || fio->ad_fsp == NULL) {
    3610           0 :                 DBG_ERR("fio/ad_fsp=NULL for [%s]\n", fsp_str_dbg(fsp));
    3611           0 :                 errno = EBADF;
    3612           0 :                 return -1;
    3613             :         }
    3614             : 
    3615             :         /* Populate the stat struct with info from the base file. */
    3616         966 :         ret = fruit_stat_base(handle, fsp->base_fsp->fsp_name, false);
    3617         966 :         if (ret == -1) {
    3618           0 :                 return -1;
    3619             :         }
    3620             : 
    3621         966 :         ad = ad_fget(talloc_tos(), handle, fio->ad_fsp, ADOUBLE_RSRC);
    3622         966 :         if (ad == NULL) {
    3623           0 :                 DBG_ERR("ad_fget [%s] failed [%s]\n",
    3624             :                         fsp_str_dbg(fio->ad_fsp), strerror(errno));
    3625           0 :                 return -1;
    3626             :         }
    3627             : 
    3628         966 :         *sbuf = fsp->base_fsp->fsp_name->st;
    3629         966 :         sbuf->st_ex_size = ad_getentrylen(ad, ADEID_RFORK);
    3630         966 :         sbuf->st_ex_ino = hash_inode(sbuf, fsp->fsp_name->stream_name);
    3631             : 
    3632         966 :         TALLOC_FREE(ad);
    3633         966 :         return 0;
    3634             : }
    3635             : 
    3636        1350 : static int fruit_fstat_rsrc(vfs_handle_struct *handle, files_struct *fsp,
    3637             :                             SMB_STRUCT_STAT *sbuf, struct fio *fio)
    3638             : {
    3639             :         int ret;
    3640             : 
    3641        1350 :         switch (fio->config->rsrc) {
    3642         384 :         case FRUIT_RSRC_STREAM:
    3643         384 :                 ret = fruit_fstat_rsrc_stream(handle, fsp, sbuf);
    3644         384 :                 break;
    3645             : 
    3646         966 :         case FRUIT_RSRC_ADFILE:
    3647         966 :                 ret = fruit_fstat_rsrc_adouble(handle, fsp, sbuf);
    3648         966 :                 break;
    3649             : 
    3650           0 :         case FRUIT_RSRC_XATTR:
    3651           0 :                 ret = fruit_fstat_rsrc_xattr(handle, fsp, sbuf);
    3652           0 :                 break;
    3653             : 
    3654           0 :         default:
    3655           0 :                 DBG_ERR("Unexpected rsrc config [%d]\n", fio->config->rsrc);
    3656           0 :                 return -1;
    3657             :         }
    3658             : 
    3659        1350 :         return ret;
    3660             : }
    3661             : 
    3662       96694 : static int fruit_fstat(vfs_handle_struct *handle, files_struct *fsp,
    3663             :                        SMB_STRUCT_STAT *sbuf)
    3664             : {
    3665       96694 :         struct fio *fio = fruit_get_complete_fio(handle, fsp);
    3666             :         int rc;
    3667             : 
    3668       96694 :         if (fio == NULL) {
    3669       91300 :                 return SMB_VFS_NEXT_FSTAT(handle, fsp, sbuf);
    3670             :         }
    3671             : 
    3672        5394 :         DBG_DEBUG("Path [%s]\n", fsp_str_dbg(fsp));
    3673             : 
    3674        5394 :         if (fio->type == ADOUBLE_META) {
    3675        4044 :                 rc = fruit_fstat_meta(handle, fsp, sbuf, fio);
    3676             :         } else {
    3677        1350 :                 rc = fruit_fstat_rsrc(handle, fsp, sbuf, fio);
    3678             :         }
    3679             : 
    3680        5394 :         if (rc == 0) {
    3681        5394 :                 sbuf->st_ex_mode &= ~S_IFMT;
    3682        5394 :                 sbuf->st_ex_mode |= S_IFREG;
    3683        5394 :                 sbuf->st_ex_blocks = sbuf->st_ex_size / STAT_ST_BLOCKSIZE + 1;
    3684             :         }
    3685             : 
    3686        5394 :         DBG_DEBUG("Path [%s] rc [%d] size [%"PRIdMAX"]\n",
    3687             :                   fsp_str_dbg(fsp), rc, (intmax_t)sbuf->st_ex_size);
    3688        5394 :         return rc;
    3689             : }
    3690             : 
    3691          18 : static NTSTATUS delete_invalid_meta_stream(
    3692             :         vfs_handle_struct *handle,
    3693             :         const struct smb_filename *smb_fname,
    3694             :         TALLOC_CTX *mem_ctx,
    3695             :         unsigned int *pnum_streams,
    3696             :         struct stream_struct **pstreams,
    3697             :         off_t size)
    3698             : {
    3699          18 :         struct smb_filename *sname = NULL;
    3700             :         NTSTATUS status;
    3701             :         int ret;
    3702             :         bool ok;
    3703             : 
    3704          18 :         ok = del_fruit_stream(mem_ctx, pnum_streams, pstreams, AFPINFO_STREAM);
    3705          18 :         if (!ok) {
    3706           0 :                 return NT_STATUS_INTERNAL_ERROR;
    3707             :         }
    3708             : 
    3709          18 :         if (size == 0) {
    3710          12 :                 return NT_STATUS_OK;
    3711             :         }
    3712             : 
    3713           6 :         status = synthetic_pathref(talloc_tos(),
    3714           6 :                                    handle->conn->cwd_fsp,
    3715           6 :                                    smb_fname->base_name,
    3716             :                                    AFPINFO_STREAM_NAME,
    3717             :                                    NULL,
    3718           6 :                                    smb_fname->twrp,
    3719             :                                    0,
    3720             :                                    &sname);
    3721           6 :         if (!NT_STATUS_IS_OK(status)) {
    3722           0 :                 return NT_STATUS_NO_MEMORY;
    3723             :         }
    3724             : 
    3725           6 :         ret = SMB_VFS_NEXT_UNLINKAT(handle,
    3726             :                         handle->conn->cwd_fsp,
    3727             :                         sname,
    3728             :                         0);
    3729           6 :         if (ret != 0) {
    3730           0 :                 DBG_ERR("Removing [%s] failed\n", smb_fname_str_dbg(sname));
    3731           0 :                 TALLOC_FREE(sname);
    3732           0 :                 return map_nt_error_from_unix(errno);
    3733             :         }
    3734             : 
    3735           6 :         TALLOC_FREE(sname);
    3736           6 :         return NT_STATUS_OK;
    3737             : }
    3738             : 
    3739        4124 : static NTSTATUS fruit_streaminfo_meta_stream(
    3740             :         vfs_handle_struct *handle,
    3741             :         struct files_struct *fsp,
    3742             :         const struct smb_filename *smb_fname,
    3743             :         TALLOC_CTX *mem_ctx,
    3744             :         unsigned int *pnum_streams,
    3745             :         struct stream_struct **pstreams)
    3746             : {
    3747        4124 :         struct stream_struct *stream = *pstreams;
    3748        4124 :         unsigned int num_streams = *pnum_streams;
    3749             :         int i;
    3750             : 
    3751        7512 :         for (i = 0; i < num_streams; i++) {
    3752        3954 :                 if (strequal_m(stream[i].name, AFPINFO_STREAM)) {
    3753         566 :                         break;
    3754             :                 }
    3755             :         }
    3756             : 
    3757        4124 :         if (i == num_streams) {
    3758        3558 :                 return NT_STATUS_OK;
    3759             :         }
    3760             : 
    3761         566 :         if (stream[i].size != AFP_INFO_SIZE) {
    3762          18 :                 DBG_ERR("Removing invalid AFPINFO_STREAM size [%jd] from [%s]\n",
    3763             :                         (intmax_t)stream[i].size, smb_fname_str_dbg(smb_fname));
    3764             : 
    3765          18 :                 return delete_invalid_meta_stream(handle,
    3766             :                                                   smb_fname,
    3767             :                                                   mem_ctx,
    3768             :                                                   pnum_streams,
    3769             :                                                   pstreams,
    3770          18 :                                                   stream[i].size);
    3771             :         }
    3772             : 
    3773             : 
    3774         548 :         return NT_STATUS_OK;
    3775             : }
    3776             : 
    3777        2616 : static NTSTATUS fruit_streaminfo_meta_netatalk(
    3778             :         vfs_handle_struct *handle,
    3779             :         struct files_struct *fsp,
    3780             :         const struct smb_filename *smb_fname,
    3781             :         TALLOC_CTX *mem_ctx,
    3782             :         unsigned int *pnum_streams,
    3783             :         struct stream_struct **pstreams)
    3784             : {
    3785        2616 :         struct stream_struct *stream = *pstreams;
    3786        2616 :         unsigned int num_streams = *pnum_streams;
    3787        2616 :         struct adouble *ad = NULL;
    3788             :         bool is_fi_empty;
    3789             :         int i;
    3790             :         bool ok;
    3791             : 
    3792             :         /* Remove the Netatalk xattr from the list */
    3793        2616 :         ok = del_fruit_stream(mem_ctx, pnum_streams, pstreams,
    3794             :                               ":" NETATALK_META_XATTR ":$DATA");
    3795        2616 :         if (!ok) {
    3796           0 :                 return NT_STATUS_NO_MEMORY;
    3797             :         }
    3798             : 
    3799             :         /*
    3800             :          * Check if there's a AFPINFO_STREAM from the VFS streams
    3801             :          * backend and if yes, remove it from the list
    3802             :          */
    3803        4330 :         for (i = 0; i < num_streams; i++) {
    3804        1722 :                 if (strequal_m(stream[i].name, AFPINFO_STREAM)) {
    3805           8 :                         break;
    3806             :                 }
    3807             :         }
    3808             : 
    3809        2616 :         if (i < num_streams) {
    3810           8 :                 DBG_WARNING("Unexpected AFPINFO_STREAM on [%s]\n",
    3811             :                             smb_fname_str_dbg(smb_fname));
    3812             : 
    3813           8 :                 ok = del_fruit_stream(mem_ctx, pnum_streams, pstreams,
    3814             :                                       AFPINFO_STREAM);
    3815           8 :                 if (!ok) {
    3816           0 :                         return NT_STATUS_INTERNAL_ERROR;
    3817             :                 }
    3818             :         }
    3819             : 
    3820        2616 :         ad = ad_get_meta_fsp(talloc_tos(), handle, smb_fname);
    3821        2616 :         if (ad == NULL) {
    3822        2422 :                 return NT_STATUS_OK;
    3823             :         }
    3824             : 
    3825         194 :         is_fi_empty = ad_empty_finderinfo(ad);
    3826         194 :         TALLOC_FREE(ad);
    3827             : 
    3828         194 :         if (is_fi_empty) {
    3829           4 :                 return NT_STATUS_OK;
    3830             :         }
    3831             : 
    3832         190 :         ok = add_fruit_stream(mem_ctx, pnum_streams, pstreams,
    3833             :                               AFPINFO_STREAM_NAME, AFP_INFO_SIZE,
    3834         190 :                               smb_roundup(handle->conn, AFP_INFO_SIZE));
    3835         190 :         if (!ok) {
    3836           0 :                 return NT_STATUS_NO_MEMORY;
    3837             :         }
    3838             : 
    3839         190 :         return NT_STATUS_OK;
    3840             : }
    3841             : 
    3842        6740 : static NTSTATUS fruit_streaminfo_meta(vfs_handle_struct *handle,
    3843             :                                       struct files_struct *fsp,
    3844             :                                       const struct smb_filename *smb_fname,
    3845             :                                       TALLOC_CTX *mem_ctx,
    3846             :                                       unsigned int *pnum_streams,
    3847             :                                       struct stream_struct **pstreams)
    3848             : {
    3849        6740 :         struct fruit_config_data *config = NULL;
    3850             :         NTSTATUS status;
    3851             : 
    3852        6740 :         SMB_VFS_HANDLE_GET_DATA(handle, config, struct fruit_config_data,
    3853             :                                 return NT_STATUS_INTERNAL_ERROR);
    3854             : 
    3855        6740 :         switch (config->meta) {
    3856        2616 :         case FRUIT_META_NETATALK:
    3857        2616 :                 status = fruit_streaminfo_meta_netatalk(handle, fsp, smb_fname,
    3858             :                                                         mem_ctx, pnum_streams,
    3859             :                                                         pstreams);
    3860        2616 :                 break;
    3861             : 
    3862        4124 :         case FRUIT_META_STREAM:
    3863        4124 :                 status = fruit_streaminfo_meta_stream(handle, fsp, smb_fname,
    3864             :                                                       mem_ctx, pnum_streams,
    3865             :                                                       pstreams);
    3866        4124 :                 break;
    3867             : 
    3868           0 :         default:
    3869           0 :                 return NT_STATUS_INTERNAL_ERROR;
    3870             :         }
    3871             : 
    3872        6740 :         return status;
    3873             : }
    3874             : 
    3875        1204 : static NTSTATUS fruit_streaminfo_rsrc_stream(
    3876             :         vfs_handle_struct *handle,
    3877             :         struct files_struct *fsp,
    3878             :         const struct smb_filename *smb_fname,
    3879             :         TALLOC_CTX *mem_ctx,
    3880             :         unsigned int *pnum_streams,
    3881             :         struct stream_struct **pstreams)
    3882             : {
    3883             :         bool ok;
    3884             : 
    3885        1204 :         ok = filter_empty_rsrc_stream(pnum_streams, pstreams);
    3886        1204 :         if (!ok) {
    3887           0 :                 DBG_ERR("Filtering resource stream failed\n");
    3888           0 :                 return NT_STATUS_INTERNAL_ERROR;
    3889             :         }
    3890        1204 :         return NT_STATUS_OK;
    3891             : }
    3892             : 
    3893           0 : static NTSTATUS fruit_streaminfo_rsrc_xattr(
    3894             :         vfs_handle_struct *handle,
    3895             :         struct files_struct *fsp,
    3896             :         const struct smb_filename *smb_fname,
    3897             :         TALLOC_CTX *mem_ctx,
    3898             :         unsigned int *pnum_streams,
    3899             :         struct stream_struct **pstreams)
    3900             : {
    3901             :         bool ok;
    3902             : 
    3903           0 :         ok = filter_empty_rsrc_stream(pnum_streams, pstreams);
    3904           0 :         if (!ok) {
    3905           0 :                 DBG_ERR("Filtering resource stream failed\n");
    3906           0 :                 return NT_STATUS_INTERNAL_ERROR;
    3907             :         }
    3908           0 :         return NT_STATUS_OK;
    3909             : }
    3910             : 
    3911        4034 : static NTSTATUS fruit_streaminfo_rsrc_adouble(
    3912             :         vfs_handle_struct *handle,
    3913             :         struct files_struct *fsp,
    3914             :         const struct smb_filename *smb_fname,
    3915             :         TALLOC_CTX *mem_ctx,
    3916             :         unsigned int *pnum_streams,
    3917             :         struct stream_struct **pstreams)
    3918             : {
    3919        4034 :         struct stream_struct *stream = *pstreams;
    3920        4034 :         unsigned int num_streams = *pnum_streams;
    3921        4034 :         struct adouble *ad = NULL;
    3922             :         bool ok;
    3923             :         size_t rlen;
    3924             :         int i;
    3925             : 
    3926             :         /*
    3927             :          * Check if there's a AFPRESOURCE_STREAM from the VFS streams backend
    3928             :          * and if yes, remove it from the list
    3929             :          */
    3930        8806 :         for (i = 0; i < num_streams; i++) {
    3931        4772 :                 if (strequal_m(stream[i].name, AFPRESOURCE_STREAM)) {
    3932           0 :                         break;
    3933             :                 }
    3934             :         }
    3935             : 
    3936        4034 :         if (i < num_streams) {
    3937           0 :                 DBG_WARNING("Unexpected AFPRESOURCE_STREAM on [%s]\n",
    3938             :                             smb_fname_str_dbg(smb_fname));
    3939             : 
    3940           0 :                 ok = del_fruit_stream(mem_ctx, pnum_streams, pstreams,
    3941             :                                       AFPRESOURCE_STREAM);
    3942           0 :                 if (!ok) {
    3943           0 :                         return NT_STATUS_INTERNAL_ERROR;
    3944             :                 }
    3945             :         }
    3946             : 
    3947        4034 :         ad = ad_get(talloc_tos(), handle, smb_fname, ADOUBLE_RSRC);
    3948        4034 :         if (ad == NULL) {
    3949        3642 :                 return NT_STATUS_OK;
    3950             :         }
    3951             : 
    3952         392 :         rlen = ad_getentrylen(ad, ADEID_RFORK);
    3953         392 :         TALLOC_FREE(ad);
    3954             : 
    3955         392 :         if (rlen == 0) {
    3956         100 :                 return NT_STATUS_OK;
    3957             :         }
    3958             : 
    3959         292 :         ok = add_fruit_stream(mem_ctx, pnum_streams, pstreams,
    3960             :                               AFPRESOURCE_STREAM_NAME, rlen,
    3961         292 :                               smb_roundup(handle->conn, rlen));
    3962         292 :         if (!ok) {
    3963           0 :                 return NT_STATUS_NO_MEMORY;
    3964             :         }
    3965             : 
    3966         292 :         return NT_STATUS_OK;
    3967             : }
    3968             : 
    3969        6740 : static NTSTATUS fruit_streaminfo_rsrc(vfs_handle_struct *handle,
    3970             :                                       struct files_struct *fsp,
    3971             :                                       const struct smb_filename *smb_fname,
    3972             :                                       TALLOC_CTX *mem_ctx,
    3973             :                                       unsigned int *pnum_streams,
    3974             :                                       struct stream_struct **pstreams)
    3975             : {
    3976        6740 :         struct fruit_config_data *config = NULL;
    3977             :         NTSTATUS status;
    3978             : 
    3979        6740 :         if (S_ISDIR(smb_fname->st.st_ex_mode)) {
    3980        1502 :                 return NT_STATUS_OK;
    3981             :         }
    3982             : 
    3983        5238 :         SMB_VFS_HANDLE_GET_DATA(handle, config, struct fruit_config_data,
    3984             :                                 return NT_STATUS_INTERNAL_ERROR);
    3985             : 
    3986        5238 :         switch (config->rsrc) {
    3987        1204 :         case FRUIT_RSRC_STREAM:
    3988        1204 :                 status = fruit_streaminfo_rsrc_stream(handle, fsp, smb_fname,
    3989             :                                                       mem_ctx, pnum_streams,
    3990             :                                                       pstreams);
    3991        1204 :                 break;
    3992             : 
    3993           0 :         case FRUIT_RSRC_XATTR:
    3994           0 :                 status = fruit_streaminfo_rsrc_xattr(handle, fsp, smb_fname,
    3995             :                                                      mem_ctx, pnum_streams,
    3996             :                                                      pstreams);
    3997           0 :                 break;
    3998             : 
    3999        4034 :         case FRUIT_RSRC_ADFILE:
    4000        4034 :                 status = fruit_streaminfo_rsrc_adouble(handle, fsp, smb_fname,
    4001             :                                                        mem_ctx, pnum_streams,
    4002             :                                                        pstreams);
    4003        4034 :                 break;
    4004             : 
    4005           0 :         default:
    4006           0 :                 return NT_STATUS_INTERNAL_ERROR;
    4007             :         }
    4008             : 
    4009        5238 :         return status;
    4010             : }
    4011             : 
    4012        6740 : static void fruit_filter_empty_streams(unsigned int *pnum_streams,
    4013             :                                        struct stream_struct **pstreams)
    4014             : {
    4015        6740 :         unsigned num_streams = *pnum_streams;
    4016        6740 :         struct stream_struct *streams = *pstreams;
    4017        6740 :         unsigned i = 0;
    4018             : 
    4019        6740 :         if (!global_fruit_config.nego_aapl) {
    4020        4992 :                 return;
    4021             :         }
    4022             : 
    4023        3680 :         while (i < num_streams) {
    4024        1932 :                 struct smb_filename smb_fname = (struct smb_filename) {
    4025        1932 :                         .stream_name = streams[i].name,
    4026             :                 };
    4027             : 
    4028        1932 :                 if (is_ntfs_default_stream_smb_fname(&smb_fname)
    4029         470 :                     || streams[i].size > 0)
    4030             :                 {
    4031        1758 :                         i++;
    4032        1758 :                         continue;
    4033             :                 }
    4034             : 
    4035         174 :                 streams[i] = streams[num_streams - 1];
    4036         174 :                 num_streams--;
    4037             :         }
    4038             : 
    4039        1748 :         *pnum_streams = num_streams;
    4040             : }
    4041             : 
    4042        6740 : static NTSTATUS fruit_fstreaminfo(vfs_handle_struct *handle,
    4043             :                                  struct files_struct *fsp,
    4044             :                                  TALLOC_CTX *mem_ctx,
    4045             :                                  unsigned int *pnum_streams,
    4046             :                                  struct stream_struct **pstreams)
    4047             : {
    4048        6740 :         struct fruit_config_data *config = NULL;
    4049        6740 :         const struct smb_filename *smb_fname = NULL;
    4050             :         NTSTATUS status;
    4051             : 
    4052        6740 :         smb_fname = fsp->fsp_name;
    4053             : 
    4054        6740 :         SMB_VFS_HANDLE_GET_DATA(handle, config, struct fruit_config_data,
    4055             :                                 return NT_STATUS_UNSUCCESSFUL);
    4056             : 
    4057        6740 :         DBG_DEBUG("Path [%s]\n", smb_fname_str_dbg(smb_fname));
    4058             : 
    4059        6740 :         status = SMB_VFS_NEXT_FSTREAMINFO(handle, fsp, mem_ctx,
    4060             :                                          pnum_streams, pstreams);
    4061        6740 :         if (!NT_STATUS_IS_OK(status)) {
    4062           0 :                 return status;
    4063             :         }
    4064             : 
    4065        6740 :         fruit_filter_empty_streams(pnum_streams, pstreams);
    4066             : 
    4067        6740 :         status = fruit_streaminfo_meta(handle, fsp, smb_fname,
    4068             :                                        mem_ctx, pnum_streams, pstreams);
    4069        6740 :         if (!NT_STATUS_IS_OK(status)) {
    4070           0 :                 return status;
    4071             :         }
    4072             : 
    4073        6740 :         status = fruit_streaminfo_rsrc(handle, fsp, smb_fname,
    4074             :                                        mem_ctx, pnum_streams, pstreams);
    4075        6740 :         if (!NT_STATUS_IS_OK(status)) {
    4076           0 :                 return status;
    4077             :         }
    4078             : 
    4079        6740 :         return NT_STATUS_OK;
    4080             : }
    4081             : 
    4082        1852 : static int fruit_fntimes(vfs_handle_struct *handle,
    4083             :                          files_struct *fsp,
    4084             :                          struct smb_file_time *ft)
    4085             : {
    4086        1852 :         int rc = 0;
    4087        1852 :         struct adouble *ad = NULL;
    4088        1852 :         struct fruit_config_data *config = NULL;
    4089             : 
    4090        1852 :         SMB_VFS_HANDLE_GET_DATA(handle, config, struct fruit_config_data,
    4091             :                                 return -1);
    4092             : 
    4093        2330 :         if ((config->meta != FRUIT_META_NETATALK) ||
    4094         478 :             is_omit_timespec(&ft->create_time))
    4095             :         {
    4096        1852 :                 return SMB_VFS_NEXT_FNTIMES(handle, fsp, ft);
    4097             :         }
    4098             : 
    4099           0 :         DBG_DEBUG("set btime for %s to %s", fsp_str_dbg(fsp),
    4100             :                   time_to_asc(convert_timespec_to_time_t(ft->create_time)));
    4101             : 
    4102           0 :         ad = ad_fget(talloc_tos(), handle, fsp, ADOUBLE_META);
    4103           0 :         if (ad == NULL) {
    4104           0 :                 goto exit;
    4105             :         }
    4106             : 
    4107           0 :         ad_setdate(ad, AD_DATE_CREATE | AD_DATE_UNIX,
    4108             :                    convert_time_t_to_uint32_t(ft->create_time.tv_sec));
    4109             : 
    4110           0 :         rc = ad_fset(handle, ad, fsp);
    4111             : 
    4112           0 : exit:
    4113             : 
    4114           0 :         TALLOC_FREE(ad);
    4115           0 :         if (rc != 0) {
    4116           0 :                 DBG_WARNING("%s\n", fsp_str_dbg(fsp));
    4117           0 :                 return -1;
    4118             :         }
    4119           0 :         return SMB_VFS_NEXT_FNTIMES(handle, fsp, ft);
    4120             : }
    4121             : 
    4122           0 : static int fruit_fallocate(struct vfs_handle_struct *handle,
    4123             :                            struct files_struct *fsp,
    4124             :                            uint32_t mode,
    4125             :                            off_t offset,
    4126             :                            off_t len)
    4127             : {
    4128           0 :         struct fio *fio = fruit_get_complete_fio(handle, fsp);
    4129             : 
    4130           0 :         if (fio == NULL) {
    4131           0 :                 return SMB_VFS_NEXT_FALLOCATE(handle, fsp, mode, offset, len);
    4132             :         }
    4133             : 
    4134             :         /* Let the pwrite code path handle it. */
    4135           0 :         errno = ENOSYS;
    4136           0 :         return -1;
    4137             : }
    4138             : 
    4139           0 : static int fruit_ftruncate_rsrc_xattr(struct vfs_handle_struct *handle,
    4140             :                                       struct files_struct *fsp,
    4141             :                                       off_t offset)
    4142             : {
    4143             : #ifdef HAVE_ATTROPEN
    4144             :         return SMB_VFS_NEXT_FTRUNCATE(handle, fsp, offset);
    4145             : #endif
    4146           0 :         return 0;
    4147             : }
    4148             : 
    4149          30 : static int fruit_ftruncate_rsrc_adouble(struct vfs_handle_struct *handle,
    4150             :                                         struct files_struct *fsp,
    4151             :                                         off_t offset)
    4152             : {
    4153          30 :         struct fio *fio = fruit_get_complete_fio(handle, fsp);
    4154             :         int rc;
    4155          30 :         struct adouble *ad = NULL;
    4156             :         off_t ad_off;
    4157             : 
    4158          30 :         if (fio == NULL || fio->ad_fsp == NULL) {
    4159           0 :                 DBG_ERR("fio/ad_fsp=NULL for [%s]\n", fsp_str_dbg(fsp));
    4160           0 :                 errno = EBADF;
    4161           0 :                 return -1;
    4162             :         }
    4163             : 
    4164          30 :         ad = ad_fget(talloc_tos(), handle, fio->ad_fsp, ADOUBLE_RSRC);
    4165          30 :         if (ad == NULL) {
    4166           0 :                 DBG_ERR("ad_fget [%s] failed [%s]\n",
    4167             :                         fsp_str_dbg(fio->ad_fsp), strerror(errno));
    4168           0 :                 return -1;
    4169             :         }
    4170             : 
    4171          30 :         ad_off = ad_getentryoff(ad, ADEID_RFORK);
    4172             : 
    4173          30 :         rc = SMB_VFS_NEXT_FTRUNCATE(handle, fio->ad_fsp, offset + ad_off);
    4174          30 :         if (rc != 0) {
    4175           0 :                 TALLOC_FREE(ad);
    4176           0 :                 return -1;
    4177             :         }
    4178             : 
    4179          30 :         ad_setentrylen(ad, ADEID_RFORK, offset);
    4180             : 
    4181          30 :         rc = ad_fset(handle, ad, fio->ad_fsp);
    4182          30 :         if (rc != 0) {
    4183           0 :                 DBG_ERR("ad_fset [%s] failed [%s]\n",
    4184             :                         fsp_str_dbg(fio->ad_fsp), strerror(errno));
    4185           0 :                 TALLOC_FREE(ad);
    4186           0 :                 return -1;
    4187             :         }
    4188             : 
    4189          30 :         TALLOC_FREE(ad);
    4190          30 :         return 0;
    4191             : }
    4192             : 
    4193          10 : static int fruit_ftruncate_rsrc_stream(struct vfs_handle_struct *handle,
    4194             :                                        struct files_struct *fsp,
    4195             :                                        off_t offset)
    4196             : {
    4197          10 :         return SMB_VFS_NEXT_FTRUNCATE(handle, fsp, offset);
    4198             : }
    4199             : 
    4200          40 : static int fruit_ftruncate_rsrc(struct vfs_handle_struct *handle,
    4201             :                                 struct files_struct *fsp,
    4202             :                                 off_t offset)
    4203             : {
    4204          40 :         struct fio *fio = fruit_get_complete_fio(handle, fsp);
    4205             :         int ret;
    4206             : 
    4207          40 :         if (fio == NULL) {
    4208           0 :                 DBG_ERR("Failed to fetch fsp extension\n");
    4209           0 :                 return -1;
    4210             :         }
    4211             : 
    4212          40 :         switch (fio->config->rsrc) {
    4213           0 :         case FRUIT_RSRC_XATTR:
    4214           0 :                 ret = fruit_ftruncate_rsrc_xattr(handle, fsp, offset);
    4215           0 :                 break;
    4216             : 
    4217          30 :         case FRUIT_RSRC_ADFILE:
    4218          30 :                 ret = fruit_ftruncate_rsrc_adouble(handle, fsp, offset);
    4219          30 :                 break;
    4220             : 
    4221          10 :         case FRUIT_RSRC_STREAM:
    4222          10 :                 ret = fruit_ftruncate_rsrc_stream(handle, fsp, offset);
    4223          10 :                 break;
    4224             : 
    4225           0 :         default:
    4226           0 :                 DBG_ERR("Unexpected rsrc config [%d]\n", fio->config->rsrc);
    4227           0 :                 return -1;
    4228             :         }
    4229             : 
    4230             : 
    4231          40 :         return ret;
    4232             : }
    4233             : 
    4234          40 : static int fruit_ftruncate_meta(struct vfs_handle_struct *handle,
    4235             :                                 struct files_struct *fsp,
    4236             :                                 off_t offset)
    4237             : {
    4238          40 :         if (offset > 60) {
    4239           8 :                 DBG_WARNING("ftruncate %s to %jd\n",
    4240             :                             fsp_str_dbg(fsp), (intmax_t)offset);
    4241             :                 /* OS X returns NT_STATUS_ALLOTTED_SPACE_EXCEEDED  */
    4242           8 :                 errno = EOVERFLOW;
    4243           8 :                 return -1;
    4244             :         }
    4245             : 
    4246             :         /* OS X returns success but does nothing  */
    4247          32 :         DBG_INFO("ignoring ftruncate %s to %jd\n",
    4248             :                  fsp_str_dbg(fsp), (intmax_t)offset);
    4249          32 :         return 0;
    4250             : }
    4251             : 
    4252         178 : static int fruit_ftruncate(struct vfs_handle_struct *handle,
    4253             :                            struct files_struct *fsp,
    4254             :                            off_t offset)
    4255             : {
    4256         178 :         struct fio *fio = fruit_get_complete_fio(handle, fsp);
    4257             :         int ret;
    4258             : 
    4259         178 :         DBG_DEBUG("Path [%s] offset [%"PRIdMAX"]\n", fsp_str_dbg(fsp),
    4260             :                   (intmax_t)offset);
    4261             : 
    4262         178 :         if (fio == NULL) {
    4263          98 :                 return SMB_VFS_NEXT_FTRUNCATE(handle, fsp, offset);
    4264             :         }
    4265             : 
    4266          80 :         if (fio->type == ADOUBLE_META) {
    4267          40 :                 ret = fruit_ftruncate_meta(handle, fsp, offset);
    4268             :         } else {
    4269          40 :                 ret = fruit_ftruncate_rsrc(handle, fsp, offset);
    4270             :         }
    4271             : 
    4272          80 :         DBG_DEBUG("Path [%s] result [%d]\n", fsp_str_dbg(fsp), ret);
    4273          80 :         return ret;
    4274             : }
    4275             : 
    4276       10598 : static NTSTATUS fruit_create_file(vfs_handle_struct *handle,
    4277             :                                   struct smb_request *req,
    4278             :                                   struct files_struct *dirfsp,
    4279             :                                   struct smb_filename *smb_fname,
    4280             :                                   uint32_t access_mask,
    4281             :                                   uint32_t share_access,
    4282             :                                   uint32_t create_disposition,
    4283             :                                   uint32_t create_options,
    4284             :                                   uint32_t file_attributes,
    4285             :                                   uint32_t oplock_request,
    4286             :                                   const struct smb2_lease *lease,
    4287             :                                   uint64_t allocation_size,
    4288             :                                   uint32_t private_flags,
    4289             :                                   struct security_descriptor *sd,
    4290             :                                   struct ea_list *ea_list,
    4291             :                                   files_struct **result,
    4292             :                                   int *pinfo,
    4293             :                                   const struct smb2_create_blobs *in_context_blobs,
    4294             :                                   struct smb2_create_blobs *out_context_blobs)
    4295             : {
    4296             :         NTSTATUS status;
    4297       10598 :         struct fruit_config_data *config = NULL;
    4298       10598 :         files_struct *fsp = NULL;
    4299       10598 :         bool internal_open = (oplock_request & INTERNAL_OPEN_ONLY);
    4300             :         int ret;
    4301             : 
    4302       10598 :         status = check_aapl(handle, req, in_context_blobs, out_context_blobs);
    4303       10598 :         if (!NT_STATUS_IS_OK(status)) {
    4304           0 :                 goto fail;
    4305             :         }
    4306             : 
    4307       10598 :         SMB_VFS_HANDLE_GET_DATA(handle, config, struct fruit_config_data,
    4308             :                                 return NT_STATUS_UNSUCCESSFUL);
    4309             : 
    4310       10598 :         if (is_apple_stream(smb_fname->stream_name) &&
    4311        3622 :             !internal_open &&
    4312        3492 :             config->convert_adouble)
    4313             :         {
    4314        3492 :                 uint32_t conv_flags  = 0;
    4315             : 
    4316        3492 :                 if (config->wipe_intentionally_left_blank_rfork) {
    4317         870 :                         conv_flags |= AD_CONV_WIPE_BLANK;
    4318             :                 }
    4319        3492 :                 if (config->delete_empty_adfiles) {
    4320         866 :                         conv_flags |= AD_CONV_DELETE;
    4321             :                 }
    4322             : 
    4323        3492 :                 ret = ad_convert(handle,
    4324             :                                  smb_fname,
    4325             :                                  macos_string_replace_map,
    4326             :                                  conv_flags);
    4327        3492 :                 if (ret != 0) {
    4328           6 :                         DBG_ERR("ad_convert(\"%s\") failed\n",
    4329             :                                 smb_fname_str_dbg(smb_fname));
    4330             :                 }
    4331             :         }
    4332             : 
    4333       10598 :         status = SMB_VFS_NEXT_CREATE_FILE(
    4334             :                 handle, req, dirfsp, smb_fname,
    4335             :                 access_mask, share_access,
    4336             :                 create_disposition, create_options,
    4337             :                 file_attributes, oplock_request,
    4338             :                 lease,
    4339             :                 allocation_size, private_flags,
    4340             :                 sd, ea_list, result,
    4341             :                 pinfo, in_context_blobs, out_context_blobs);
    4342       10598 :         if (!NT_STATUS_IS_OK(status)) {
    4343        1756 :                 return status;
    4344             :         }
    4345             : 
    4346        8842 :         fsp = *result;
    4347             : 
    4348        8842 :         if (global_fruit_config.nego_aapl) {
    4349        2248 :                 if (config->posix_rename && fsp->fsp_flags.is_directory) {
    4350             :                         /*
    4351             :                          * Enable POSIX directory rename behaviour
    4352             :                          */
    4353         398 :                         fsp->posix_flags |= FSP_POSIX_FLAGS_RENAME;
    4354             :                 }
    4355             :         }
    4356             : 
    4357             :         /*
    4358             :          * If this is a plain open for existing files, opening an 0
    4359             :          * byte size resource fork MUST fail with
    4360             :          * NT_STATUS_OBJECT_NAME_NOT_FOUND.
    4361             :          *
    4362             :          * Cf the vfs_fruit torture tests in test_rfork_create().
    4363             :          */
    4364        8842 :         if (global_fruit_config.nego_aapl &&
    4365        1478 :             create_disposition == FILE_OPEN &&
    4366        2504 :             smb_fname->st.st_ex_size == 0 &&
    4367        1026 :             is_named_stream(smb_fname))
    4368             :         {
    4369         290 :                 status = NT_STATUS_OBJECT_NAME_NOT_FOUND;
    4370         290 :                 goto fail;
    4371             :         }
    4372             : 
    4373        8552 :         if (is_named_stream(smb_fname) || fsp->fsp_flags.is_directory) {
    4374        5310 :                 return status;
    4375             :         }
    4376             : 
    4377        3242 :         if ((config->locking == FRUIT_LOCKING_NETATALK) &&
    4378        1274 :             (fsp->op != NULL) &&
    4379        1270 :             !fsp->fsp_flags.is_pathref)
    4380             :         {
    4381         736 :                 status = fruit_check_access(
    4382             :                         handle, *result,
    4383             :                         access_mask,
    4384             :                         share_access);
    4385         736 :                 if (!NT_STATUS_IS_OK(status)) {
    4386           2 :                         goto fail;
    4387             :                 }
    4388             :         }
    4389             : 
    4390        3240 :         return status;
    4391             : 
    4392         292 : fail:
    4393         292 :         DEBUG(10, ("fruit_create_file: %s\n", nt_errstr(status)));
    4394             : 
    4395         292 :         if (fsp) {
    4396         292 :                 close_file_free(req, &fsp, ERROR_CLOSE);
    4397         292 :                 *result = NULL;
    4398             :         }
    4399             : 
    4400         292 :         return status;
    4401             : }
    4402             : 
    4403        1334 : static NTSTATUS fruit_freaddir_attr(struct vfs_handle_struct *handle,
    4404             :                                     struct files_struct *fsp,
    4405             :                                     TALLOC_CTX *mem_ctx,
    4406             :                                     struct readdir_attr_data **pattr_data)
    4407             : {
    4408        1334 :         struct fruit_config_data *config = NULL;
    4409             :         struct readdir_attr_data *attr_data;
    4410        1334 :         uint32_t conv_flags  = 0;
    4411             :         NTSTATUS status;
    4412             :         int ret;
    4413             : 
    4414        1334 :         SMB_VFS_HANDLE_GET_DATA(handle, config,
    4415             :                                 struct fruit_config_data,
    4416             :                                 return NT_STATUS_UNSUCCESSFUL);
    4417             : 
    4418        1334 :         if (!global_fruit_config.nego_aapl) {
    4419        1030 :                 return SMB_VFS_NEXT_FREADDIR_ATTR(handle,
    4420             :                                                   fsp,
    4421             :                                                   mem_ctx,
    4422             :                                                   pattr_data);
    4423             :         }
    4424             : 
    4425         304 :         DBG_DEBUG("Path [%s]\n", fsp_str_dbg(fsp));
    4426             : 
    4427         304 :         if (config->convert_adouble) {
    4428         304 :                 if (config->wipe_intentionally_left_blank_rfork) {
    4429          92 :                         conv_flags |= AD_CONV_WIPE_BLANK;
    4430             :                 }
    4431         304 :                 if (config->delete_empty_adfiles) {
    4432          76 :                         conv_flags |= AD_CONV_DELETE;
    4433             :                 }
    4434             : 
    4435         304 :                 ret = ad_convert(handle,
    4436         304 :                                  fsp->fsp_name,
    4437             :                                  macos_string_replace_map,
    4438             :                                  conv_flags);
    4439         304 :                 if (ret != 0) {
    4440           0 :                         DBG_ERR("ad_convert(\"%s\") failed\n",
    4441             :                                 fsp_str_dbg(fsp));
    4442             :                 }
    4443             :         }
    4444             : 
    4445         304 :         *pattr_data = talloc_zero(mem_ctx, struct readdir_attr_data);
    4446         304 :         if (*pattr_data == NULL) {
    4447           0 :                 return NT_STATUS_NO_MEMORY;
    4448             :         }
    4449         304 :         attr_data = *pattr_data;
    4450         304 :         attr_data->type = RDATTR_AAPL;
    4451             : 
    4452             :         /*
    4453             :          * Mac metadata: compressed FinderInfo, resource fork length
    4454             :          * and creation date
    4455             :          */
    4456         304 :         status = readdir_attr_macmeta(handle, fsp->fsp_name, attr_data);
    4457         304 :         if (!NT_STATUS_IS_OK(status)) {
    4458             :                 /*
    4459             :                  * Error handling is tricky: if we return failure from
    4460             :                  * this function, the corresponding directory entry
    4461             :                  * will to be passed to the client, so we really just
    4462             :                  * want to error out on fatal errors.
    4463             :                  */
    4464           0 :                 if  (!NT_STATUS_EQUAL(status, NT_STATUS_ACCESS_DENIED)) {
    4465           0 :                         goto fail;
    4466             :                 }
    4467             :         }
    4468             : 
    4469             :         /*
    4470             :          * UNIX mode
    4471             :          */
    4472         304 :         if (config->unix_info_enabled) {
    4473         304 :                 attr_data->attr_data.aapl.unix_mode =
    4474         304 :                         fsp->fsp_name->st.st_ex_mode;
    4475             :         }
    4476             : 
    4477             :         /*
    4478             :          * max_access
    4479             :          */
    4480         304 :         if (!config->readdir_attr_max_access) {
    4481           0 :                 attr_data->attr_data.aapl.max_access = FILE_GENERIC_ALL;
    4482             :         } else {
    4483         304 :                 status = smbd_calculate_access_mask_fsp(fsp->conn->cwd_fsp,
    4484             :                         fsp,
    4485             :                         false,
    4486             :                         SEC_FLAG_MAXIMUM_ALLOWED,
    4487             :                         &attr_data->attr_data.aapl.max_access);
    4488         304 :                 if (!NT_STATUS_IS_OK(status)) {
    4489           0 :                         goto fail;
    4490             :                 }
    4491             :         }
    4492             : 
    4493         304 :         return NT_STATUS_OK;
    4494             : 
    4495           0 : fail:
    4496           0 :         DBG_WARNING("Path [%s], error: %s\n", fsp_str_dbg(fsp),
    4497             :                    nt_errstr(status));
    4498           0 :         TALLOC_FREE(*pattr_data);
    4499           0 :         return status;
    4500             : }
    4501             : 
    4502       17416 : static NTSTATUS fruit_fget_nt_acl(vfs_handle_struct *handle,
    4503             :                                   files_struct *fsp,
    4504             :                                   uint32_t security_info,
    4505             :                                   TALLOC_CTX *mem_ctx,
    4506             :                                   struct security_descriptor **ppdesc)
    4507             : {
    4508             :         NTSTATUS status;
    4509             :         struct security_ace ace;
    4510             :         struct dom_sid sid;
    4511             :         struct fruit_config_data *config;
    4512             : 
    4513       17416 :         SMB_VFS_HANDLE_GET_DATA(handle, config,
    4514             :                                 struct fruit_config_data,
    4515             :                                 return NT_STATUS_UNSUCCESSFUL);
    4516             : 
    4517       17416 :         status = SMB_VFS_NEXT_FGET_NT_ACL(handle, fsp, security_info,
    4518             :                                           mem_ctx, ppdesc);
    4519       17416 :         if (!NT_STATUS_IS_OK(status)) {
    4520           0 :                 return status;
    4521             :         }
    4522             : 
    4523             :         /*
    4524             :          * Add MS NFS style ACEs with uid, gid and mode
    4525             :          */
    4526       17416 :         if (!global_fruit_config.nego_aapl) {
    4527       12702 :                 return NT_STATUS_OK;
    4528             :         }
    4529        4714 :         if (!config->unix_info_enabled) {
    4530           0 :                 return NT_STATUS_OK;
    4531             :         }
    4532             : 
    4533             :         /* First remove any existing ACE's with NFS style mode/uid/gid SIDs. */
    4534        4714 :         status = remove_virtual_nfs_aces(*ppdesc);
    4535        4714 :         if (!NT_STATUS_IS_OK(status)) {
    4536           0 :                 DBG_WARNING("failed to remove MS NFS style ACEs\n");
    4537           0 :                 return status;
    4538             :         }
    4539             : 
    4540             :         /* MS NFS style mode */
    4541        4714 :         sid_compose(&sid, &global_sid_Unix_NFS_Mode, fsp->fsp_name->st.st_ex_mode);
    4542        4714 :         init_sec_ace(&ace, &sid, SEC_ACE_TYPE_ACCESS_DENIED, 0, 0);
    4543        4714 :         status = security_descriptor_dacl_add(*ppdesc, &ace);
    4544        4714 :         if (!NT_STATUS_IS_OK(status)) {
    4545           0 :                 DEBUG(1,("failed to add MS NFS style ACE\n"));
    4546           0 :                 return status;
    4547             :         }
    4548             : 
    4549             :         /* MS NFS style uid */
    4550        4714 :         sid_compose(&sid, &global_sid_Unix_NFS_Users, fsp->fsp_name->st.st_ex_uid);
    4551        4714 :         init_sec_ace(&ace, &sid, SEC_ACE_TYPE_ACCESS_DENIED, 0, 0);
    4552        4714 :         status = security_descriptor_dacl_add(*ppdesc, &ace);
    4553        4714 :         if (!NT_STATUS_IS_OK(status)) {
    4554           0 :                 DEBUG(1,("failed to add MS NFS style ACE\n"));
    4555           0 :                 return status;
    4556             :         }
    4557             : 
    4558             :         /* MS NFS style gid */
    4559        4714 :         sid_compose(&sid, &global_sid_Unix_NFS_Groups, fsp->fsp_name->st.st_ex_gid);
    4560        4714 :         init_sec_ace(&ace, &sid, SEC_ACE_TYPE_ACCESS_DENIED, 0, 0);
    4561        4714 :         status = security_descriptor_dacl_add(*ppdesc, &ace);
    4562        4714 :         if (!NT_STATUS_IS_OK(status)) {
    4563           0 :                 DEBUG(1,("failed to add MS NFS style ACE\n"));
    4564           0 :                 return status;
    4565             :         }
    4566             : 
    4567        4714 :         return NT_STATUS_OK;
    4568             : }
    4569             : 
    4570        1158 : static NTSTATUS fruit_fset_nt_acl(vfs_handle_struct *handle,
    4571             :                                   files_struct *fsp,
    4572             :                                   uint32_t security_info_sent,
    4573             :                                   const struct security_descriptor *orig_psd)
    4574             : {
    4575             :         NTSTATUS status;
    4576             :         bool do_chmod;
    4577        1158 :         mode_t ms_nfs_mode = 0;
    4578             :         int result;
    4579        1158 :         struct security_descriptor *psd = NULL;
    4580        1158 :         uint32_t orig_num_aces = 0;
    4581             : 
    4582        1158 :         if (orig_psd->dacl != NULL) {
    4583        1158 :                 orig_num_aces = orig_psd->dacl->num_aces;
    4584             :         }
    4585             : 
    4586        1158 :         psd = security_descriptor_copy(talloc_tos(), orig_psd);
    4587        1158 :         if (psd == NULL) {
    4588           0 :                 return NT_STATUS_NO_MEMORY;
    4589             :         }
    4590             : 
    4591        1158 :         DBG_DEBUG("fruit_fset_nt_acl: %s\n", fsp_str_dbg(fsp));
    4592             : 
    4593        1158 :         status = check_ms_nfs(handle, fsp, psd, &ms_nfs_mode, &do_chmod);
    4594        1158 :         if (!NT_STATUS_IS_OK(status)) {
    4595           0 :                 DEBUG(1, ("fruit_fset_nt_acl: check_ms_nfs failed%s\n", fsp_str_dbg(fsp)));
    4596           0 :                 TALLOC_FREE(psd);
    4597           0 :                 return status;
    4598             :         }
    4599             : 
    4600             :         /*
    4601             :          * If only ms_nfs ACE entries were sent, ensure we set the DACL
    4602             :          * sent/present flags correctly now we've removed them.
    4603             :          */
    4604             : 
    4605        1158 :         if (orig_num_aces != 0) {
    4606             :                 /*
    4607             :                  * Are there any ACE's left ?
    4608             :                  */
    4609        1158 :                 if (psd->dacl->num_aces == 0) {
    4610             :                         /* No - clear the DACL sent/present flags. */
    4611           0 :                         security_info_sent &= ~SECINFO_DACL;
    4612           0 :                         psd->type &= ~SEC_DESC_DACL_PRESENT;
    4613             :                 }
    4614             :         }
    4615             : 
    4616        1158 :         status = SMB_VFS_NEXT_FSET_NT_ACL(handle, fsp, security_info_sent, psd);
    4617        1158 :         if (!NT_STATUS_IS_OK(status)) {
    4618           0 :                 DEBUG(1, ("fruit_fset_nt_acl: SMB_VFS_NEXT_FSET_NT_ACL failed%s\n", fsp_str_dbg(fsp)));
    4619           0 :                 TALLOC_FREE(psd);
    4620           0 :                 return status;
    4621             :         }
    4622             : 
    4623        1158 :         if (do_chmod) {
    4624           8 :                 result = SMB_VFS_FCHMOD(fsp, ms_nfs_mode);
    4625           8 :                 if (result != 0) {
    4626           0 :                         DBG_WARNING("%s, result: %d, %04o error %s\n",
    4627             :                                 fsp_str_dbg(fsp),
    4628             :                                 result,
    4629             :                                 (unsigned)ms_nfs_mode,
    4630             :                                 strerror(errno));
    4631           0 :                         status = map_nt_error_from_unix(errno);
    4632           0 :                         TALLOC_FREE(psd);
    4633           0 :                         return status;
    4634             :                 }
    4635             :         }
    4636             : 
    4637        1158 :         TALLOC_FREE(psd);
    4638        1158 :         return NT_STATUS_OK;
    4639             : }
    4640             : 
    4641             : static struct vfs_offload_ctx *fruit_offload_ctx;
    4642             : 
    4643             : struct fruit_offload_read_state {
    4644             :         struct vfs_handle_struct *handle;
    4645             :         struct tevent_context *ev;
    4646             :         files_struct *fsp;
    4647             :         uint32_t fsctl;
    4648             :         uint32_t flags;
    4649             :         uint64_t xferlen;
    4650             :         DATA_BLOB token;
    4651             : };
    4652             : 
    4653             : static void fruit_offload_read_done(struct tevent_req *subreq);
    4654             : 
    4655          40 : static struct tevent_req *fruit_offload_read_send(
    4656             :         TALLOC_CTX *mem_ctx,
    4657             :         struct tevent_context *ev,
    4658             :         struct vfs_handle_struct *handle,
    4659             :         files_struct *fsp,
    4660             :         uint32_t fsctl,
    4661             :         uint32_t ttl,
    4662             :         off_t offset,
    4663             :         size_t to_copy)
    4664             : {
    4665          40 :         struct tevent_req *req = NULL;
    4666          40 :         struct tevent_req *subreq = NULL;
    4667          40 :         struct fruit_offload_read_state *state = NULL;
    4668             : 
    4669          40 :         req = tevent_req_create(mem_ctx, &state,
    4670             :                                 struct fruit_offload_read_state);
    4671          40 :         if (req == NULL) {
    4672           0 :                 return NULL;
    4673             :         }
    4674          40 :         *state = (struct fruit_offload_read_state) {
    4675             :                 .handle = handle,
    4676             :                 .ev = ev,
    4677             :                 .fsp = fsp,
    4678             :                 .fsctl = fsctl,
    4679             :         };
    4680             : 
    4681          40 :         subreq = SMB_VFS_NEXT_OFFLOAD_READ_SEND(mem_ctx, ev, handle, fsp,
    4682             :                                                 fsctl, ttl, offset, to_copy);
    4683          40 :         if (tevent_req_nomem(subreq, req)) {
    4684           0 :                 return tevent_req_post(req, ev);
    4685             :         }
    4686          40 :         tevent_req_set_callback(subreq, fruit_offload_read_done, req);
    4687          40 :         return req;
    4688             : }
    4689             : 
    4690          40 : static void fruit_offload_read_done(struct tevent_req *subreq)
    4691             : {
    4692          40 :         struct tevent_req *req = tevent_req_callback_data(
    4693             :                 subreq, struct tevent_req);
    4694          40 :         struct fruit_offload_read_state *state = tevent_req_data(
    4695             :                 req, struct fruit_offload_read_state);
    4696             :         NTSTATUS status;
    4697             : 
    4698          40 :         status = SMB_VFS_NEXT_OFFLOAD_READ_RECV(subreq,
    4699             :                                                 state->handle,
    4700             :                                                 state,
    4701             :                                                 &state->flags,
    4702             :                                                 &state->xferlen,
    4703             :                                                 &state->token);
    4704          40 :         TALLOC_FREE(subreq);
    4705          40 :         if (tevent_req_nterror(req, status)) {
    4706           0 :                 return;
    4707             :         }
    4708             : 
    4709          40 :         if (state->fsctl != FSCTL_SRV_REQUEST_RESUME_KEY) {
    4710           0 :                 tevent_req_done(req);
    4711           0 :                 return;
    4712             :         }
    4713             : 
    4714          40 :         status = vfs_offload_token_ctx_init(state->fsp->conn->sconn->client,
    4715             :                                             &fruit_offload_ctx);
    4716          40 :         if (tevent_req_nterror(req, status)) {
    4717           0 :                 return;
    4718             :         }
    4719             : 
    4720          40 :         status = vfs_offload_token_db_store_fsp(fruit_offload_ctx,
    4721          40 :                                                 state->fsp,
    4722          40 :                                                 &state->token);
    4723          40 :         if (tevent_req_nterror(req, status)) {
    4724           0 :                 return;
    4725             :         }
    4726             : 
    4727          40 :         tevent_req_done(req);
    4728          40 :         return;
    4729             : }
    4730             : 
    4731          40 : static NTSTATUS fruit_offload_read_recv(struct tevent_req *req,
    4732             :                                         struct vfs_handle_struct *handle,
    4733             :                                         TALLOC_CTX *mem_ctx,
    4734             :                                         uint32_t *flags,
    4735             :                                         uint64_t *xferlen,
    4736             :                                         DATA_BLOB *token)
    4737             : {
    4738          40 :         struct fruit_offload_read_state *state = tevent_req_data(
    4739             :                 req, struct fruit_offload_read_state);
    4740             :         NTSTATUS status;
    4741             : 
    4742          40 :         if (tevent_req_is_nterror(req, &status)) {
    4743           0 :                 tevent_req_received(req);
    4744           0 :                 return status;
    4745             :         }
    4746             : 
    4747          40 :         *flags = state->flags;
    4748          40 :         *xferlen = state->xferlen;
    4749          40 :         token->length = state->token.length;
    4750          40 :         token->data = talloc_move(mem_ctx, &state->token.data);
    4751             : 
    4752          40 :         tevent_req_received(req);
    4753          40 :         return NT_STATUS_OK;
    4754             : }
    4755             : 
    4756             : struct fruit_offload_write_state {
    4757             :         struct vfs_handle_struct *handle;
    4758             :         off_t copied;
    4759             :         struct files_struct *src_fsp;
    4760             :         struct files_struct *dst_fsp;
    4761             :         bool is_copyfile;
    4762             : };
    4763             : 
    4764             : static void fruit_offload_write_done(struct tevent_req *subreq);
    4765          40 : static struct tevent_req *fruit_offload_write_send(struct vfs_handle_struct *handle,
    4766             :                                                 TALLOC_CTX *mem_ctx,
    4767             :                                                 struct tevent_context *ev,
    4768             :                                                 uint32_t fsctl,
    4769             :                                                 DATA_BLOB *token,
    4770             :                                                 off_t transfer_offset,
    4771             :                                                 struct files_struct *dest_fsp,
    4772             :                                                 off_t dest_off,
    4773             :                                                 off_t num)
    4774             : {
    4775             :         struct tevent_req *req, *subreq;
    4776             :         struct fruit_offload_write_state *state;
    4777             :         NTSTATUS status;
    4778             :         struct fruit_config_data *config;
    4779          40 :         off_t src_off = transfer_offset;
    4780          40 :         files_struct *src_fsp = NULL;
    4781          40 :         off_t to_copy = num;
    4782          40 :         bool copyfile_enabled = false;
    4783             : 
    4784          40 :         DEBUG(10,("soff: %ju, doff: %ju, len: %ju\n",
    4785             :                   (uintmax_t)src_off, (uintmax_t)dest_off, (uintmax_t)num));
    4786             : 
    4787          40 :         SMB_VFS_HANDLE_GET_DATA(handle, config,
    4788             :                                 struct fruit_config_data,
    4789             :                                 return NULL);
    4790             : 
    4791          40 :         req = tevent_req_create(mem_ctx, &state,
    4792             :                                 struct fruit_offload_write_state);
    4793          40 :         if (req == NULL) {
    4794           0 :                 return NULL;
    4795             :         }
    4796          40 :         state->handle = handle;
    4797          40 :         state->dst_fsp = dest_fsp;
    4798             : 
    4799          40 :         switch (fsctl) {
    4800          40 :         case FSCTL_SRV_COPYCHUNK:
    4801             :         case FSCTL_SRV_COPYCHUNK_WRITE:
    4802          40 :                 copyfile_enabled = config->copyfile_enabled;
    4803          40 :                 break;
    4804           0 :         default:
    4805           0 :                 break;
    4806             :         }
    4807             : 
    4808             :         /*
    4809             :          * Check if this a OS X copyfile style copychunk request with
    4810             :          * a requested chunk count of 0 that was translated to a
    4811             :          * offload_write_send VFS call overloading the parameters src_off
    4812             :          * = dest_off = num = 0.
    4813             :          */
    4814          40 :         if (copyfile_enabled && num == 0 && src_off == 0 && dest_off == 0) {
    4815           8 :                 status = vfs_offload_token_db_fetch_fsp(
    4816             :                         fruit_offload_ctx, token, &src_fsp);
    4817           8 :                 if (tevent_req_nterror(req, status)) {
    4818           0 :                         return tevent_req_post(req, ev);
    4819             :                 }
    4820           8 :                 state->src_fsp = src_fsp;
    4821             : 
    4822           8 :                 status = vfs_stat_fsp(src_fsp);
    4823           8 :                 if (tevent_req_nterror(req, status)) {
    4824           0 :                         return tevent_req_post(req, ev);
    4825             :                 }
    4826             : 
    4827           8 :                 to_copy = src_fsp->fsp_name->st.st_ex_size;
    4828           8 :                 state->is_copyfile = true;
    4829             :         }
    4830             : 
    4831          40 :         subreq = SMB_VFS_NEXT_OFFLOAD_WRITE_SEND(handle,
    4832             :                                               mem_ctx,
    4833             :                                               ev,
    4834             :                                               fsctl,
    4835             :                                               token,
    4836             :                                               transfer_offset,
    4837             :                                               dest_fsp,
    4838             :                                               dest_off,
    4839             :                                               to_copy);
    4840          40 :         if (tevent_req_nomem(subreq, req)) {
    4841           0 :                 return tevent_req_post(req, ev);
    4842             :         }
    4843             : 
    4844          40 :         tevent_req_set_callback(subreq, fruit_offload_write_done, req);
    4845          40 :         return req;
    4846             : }
    4847             : 
    4848          40 : static void fruit_offload_write_done(struct tevent_req *subreq)
    4849             : {
    4850          40 :         struct tevent_req *req = tevent_req_callback_data(
    4851             :                 subreq, struct tevent_req);
    4852          40 :         struct fruit_offload_write_state *state = tevent_req_data(
    4853             :                 req, struct fruit_offload_write_state);
    4854             :         NTSTATUS status;
    4855          40 :         unsigned int num_streams = 0;
    4856          40 :         struct stream_struct *streams = NULL;
    4857             :         unsigned int i;
    4858          40 :         struct smb_filename *src_fname_tmp = NULL;
    4859          40 :         struct smb_filename *dst_fname_tmp = NULL;
    4860             : 
    4861          40 :         status = SMB_VFS_NEXT_OFFLOAD_WRITE_RECV(state->handle,
    4862             :                                               subreq,
    4863             :                                               &state->copied);
    4864          40 :         TALLOC_FREE(subreq);
    4865          40 :         if (tevent_req_nterror(req, status)) {
    4866          32 :                 return;
    4867             :         }
    4868             : 
    4869          40 :         if (!state->is_copyfile) {
    4870          32 :                 tevent_req_done(req);
    4871          32 :                 return;
    4872             :         }
    4873             : 
    4874             :         /*
    4875             :          * Now copy all remaining streams. We know the share supports
    4876             :          * streams, because we're in vfs_fruit. We don't do this async
    4877             :          * because streams are few and small.
    4878             :          */
    4879           8 :         status = vfs_fstreaminfo(state->src_fsp,
    4880             :                                 req, &num_streams, &streams);
    4881           8 :         if (tevent_req_nterror(req, status)) {
    4882           0 :                 return;
    4883             :         }
    4884             : 
    4885           8 :         if (num_streams == 1) {
    4886             :                 /* There is always one stream, ::$DATA. */
    4887           0 :                 tevent_req_done(req);
    4888           0 :                 return;
    4889             :         }
    4890             : 
    4891          32 :         for (i = 0; i < num_streams; i++) {
    4892          24 :                 DEBUG(10, ("%s: stream: '%s'/%zu\n",
    4893             :                           __func__, streams[i].name, (size_t)streams[i].size));
    4894             : 
    4895          24 :                 src_fname_tmp = synthetic_smb_fname(
    4896             :                         req,
    4897          24 :                         state->src_fsp->fsp_name->base_name,
    4898          24 :                         streams[i].name,
    4899             :                         NULL,
    4900          24 :                         state->src_fsp->fsp_name->twrp,
    4901          24 :                         state->src_fsp->fsp_name->flags);
    4902          24 :                 if (tevent_req_nomem(src_fname_tmp, req)) {
    4903           0 :                         return;
    4904             :                 }
    4905             : 
    4906          24 :                 if (is_ntfs_default_stream_smb_fname(src_fname_tmp)) {
    4907           8 :                         TALLOC_FREE(src_fname_tmp);
    4908           8 :                         continue;
    4909             :                 }
    4910             : 
    4911          16 :                 dst_fname_tmp = synthetic_smb_fname(
    4912             :                         req,
    4913          16 :                         state->dst_fsp->fsp_name->base_name,
    4914          16 :                         streams[i].name,
    4915             :                         NULL,
    4916          16 :                         state->dst_fsp->fsp_name->twrp,
    4917          16 :                         state->dst_fsp->fsp_name->flags);
    4918          16 :                 if (tevent_req_nomem(dst_fname_tmp, req)) {
    4919           0 :                         TALLOC_FREE(src_fname_tmp);
    4920           0 :                         return;
    4921             :                 }
    4922             : 
    4923          16 :                 status = copy_file(req,
    4924          16 :                                    state->handle->conn,
    4925             :                                    src_fname_tmp,
    4926             :                                    dst_fname_tmp,
    4927             :                                    FILE_CREATE);
    4928          16 :                 if (!NT_STATUS_IS_OK(status)) {
    4929           0 :                         DEBUG(1, ("%s: copy %s to %s failed: %s\n", __func__,
    4930             :                                   smb_fname_str_dbg(src_fname_tmp),
    4931             :                                   smb_fname_str_dbg(dst_fname_tmp),
    4932             :                                   nt_errstr(status)));
    4933           0 :                         TALLOC_FREE(src_fname_tmp);
    4934           0 :                         TALLOC_FREE(dst_fname_tmp);
    4935           0 :                         tevent_req_nterror(req, status);
    4936           0 :                         return;
    4937             :                 }
    4938             : 
    4939          16 :                 TALLOC_FREE(src_fname_tmp);
    4940          16 :                 TALLOC_FREE(dst_fname_tmp);
    4941             :         }
    4942             : 
    4943           8 :         TALLOC_FREE(streams);
    4944           8 :         TALLOC_FREE(src_fname_tmp);
    4945           8 :         TALLOC_FREE(dst_fname_tmp);
    4946           8 :         tevent_req_done(req);
    4947             : }
    4948             : 
    4949          40 : static NTSTATUS fruit_offload_write_recv(struct vfs_handle_struct *handle,
    4950             :                                       struct tevent_req *req,
    4951             :                                       off_t *copied)
    4952             : {
    4953          40 :         struct fruit_offload_write_state *state = tevent_req_data(
    4954             :                 req, struct fruit_offload_write_state);
    4955             :         NTSTATUS status;
    4956             : 
    4957          40 :         if (tevent_req_is_nterror(req, &status)) {
    4958           0 :                 DEBUG(1, ("server side copy chunk failed: %s\n",
    4959             :                           nt_errstr(status)));
    4960           0 :                 *copied = 0;
    4961           0 :                 tevent_req_received(req);
    4962           0 :                 return status;
    4963             :         }
    4964             : 
    4965          40 :         *copied = state->copied;
    4966          40 :         tevent_req_received(req);
    4967             : 
    4968          40 :         return NT_STATUS_OK;
    4969             : }
    4970             : 
    4971           2 : static char *fruit_get_bandsize_line(char **lines, int numlines)
    4972             : {
    4973             :         static regex_t re;
    4974             :         static bool re_initialized = false;
    4975             :         int i;
    4976             :         int ret;
    4977             : 
    4978           2 :         if (!re_initialized) {
    4979           2 :                 ret = regcomp(&re, "^[[:blank:]]*<key>band-size</key>$", 0);
    4980           2 :                 if (ret != 0) {
    4981           0 :                         return NULL;
    4982             :                 }
    4983           2 :                 re_initialized = true;
    4984             :         }
    4985             : 
    4986           4 :         for (i = 0; i < numlines; i++) {
    4987             :                 regmatch_t matches[1];
    4988             : 
    4989           4 :                 ret = regexec(&re, lines[i], 1, matches, 0);
    4990           4 :                 if (ret == 0) {
    4991             :                         /*
    4992             :                          * Check if the match was on the last line, sa we want
    4993             :                          * the subsequent line.
    4994             :                          */
    4995           2 :                         if (i + 1 == numlines) {
    4996           2 :                                 return NULL;
    4997             :                         }
    4998           2 :                         return lines[i + 1];
    4999             :                 }
    5000           2 :                 if (ret != REG_NOMATCH) {
    5001           0 :                         return NULL;
    5002             :                 }
    5003             :         }
    5004             : 
    5005           0 :         return NULL;
    5006             : }
    5007             : 
    5008           2 : static bool fruit_get_bandsize_from_line(char *line, size_t *_band_size)
    5009             : {
    5010             :         static regex_t re;
    5011             :         static bool re_initialized = false;
    5012             :         regmatch_t matches[2];
    5013             :         uint64_t band_size;
    5014             :         int ret;
    5015             :         bool ok;
    5016             : 
    5017           2 :         if (!re_initialized) {
    5018           2 :                 ret = regcomp(&re,
    5019             :                               "^[[:blank:]]*"
    5020             :                               "<integer>\\([[:digit:]]*\\)</integer>$",
    5021             :                               0);
    5022           2 :                 if (ret != 0) {
    5023           0 :                         return false;
    5024             :                 }
    5025           2 :                 re_initialized = true;
    5026             :         }
    5027             : 
    5028           2 :         ret = regexec(&re, line, 2, matches, 0);
    5029           2 :         if (ret != 0) {
    5030           0 :                 DBG_ERR("regex failed [%s]\n", line);
    5031           0 :                 return false;
    5032             :         }
    5033             : 
    5034           2 :         line[matches[1].rm_eo] = '\0';
    5035             : 
    5036           2 :         ok = conv_str_u64(&line[matches[1].rm_so], &band_size);
    5037           2 :         if (!ok) {
    5038           0 :                 return false;
    5039             :         }
    5040           2 :         *_band_size = (size_t)band_size;
    5041           2 :         return true;
    5042             : }
    5043             : 
    5044             : /*
    5045             :  * This reads and parses an Info.plist from a TM sparsebundle looking for the
    5046             :  * "band-size" key and value.
    5047             :  */
    5048           2 : static bool fruit_get_bandsize(vfs_handle_struct *handle,
    5049             :                                const char *dir,
    5050             :                                size_t *band_size)
    5051             : {
    5052             : #define INFO_PLIST_MAX_SIZE 64*1024
    5053           2 :         char *plist = NULL;
    5054           2 :         struct smb_filename *smb_fname = NULL;
    5055           2 :         files_struct *fsp = NULL;
    5056           2 :         uint8_t *file_data = NULL;
    5057           2 :         char **lines = NULL;
    5058           2 :         char *band_size_line = NULL;
    5059             :         size_t plist_file_size;
    5060             :         ssize_t nread;
    5061             :         int numlines;
    5062             :         int ret;
    5063           2 :         bool ok = false;
    5064             :         NTSTATUS status;
    5065             : 
    5066           2 :         plist = talloc_asprintf(talloc_tos(),
    5067             :                                 "%s/%s/Info.plist",
    5068           2 :                                 handle->conn->connectpath,
    5069             :                                 dir);
    5070           2 :         if (plist == NULL) {
    5071           0 :                 ok = false;
    5072           0 :                 goto out;
    5073             :         }
    5074             : 
    5075           2 :         smb_fname = synthetic_smb_fname(talloc_tos(),
    5076             :                                         plist,
    5077             :                                         NULL,
    5078             :                                         NULL,
    5079             :                                         0,
    5080             :                                         0);
    5081           2 :         if (smb_fname == NULL) {
    5082           0 :                 ok = false;
    5083           0 :                 goto out;
    5084             :         }
    5085             : 
    5086           2 :         ret = SMB_VFS_NEXT_LSTAT(handle, smb_fname);
    5087           2 :         if (ret != 0) {
    5088           0 :                 DBG_INFO("Ignoring Sparsebundle without Info.plist [%s]\n", dir);
    5089           0 :                 ok = true;
    5090           0 :                 goto out;
    5091             :         }
    5092             : 
    5093           2 :         plist_file_size = smb_fname->st.st_ex_size;
    5094             : 
    5095           2 :         if (plist_file_size > INFO_PLIST_MAX_SIZE) {
    5096           0 :                 DBG_INFO("%s is too large, ignoring\n", plist);
    5097           0 :                 ok = true;
    5098           0 :                 goto out;
    5099             :         }
    5100             : 
    5101           2 :         status = SMB_VFS_NEXT_CREATE_FILE(
    5102             :                 handle,                         /* conn */
    5103             :                 NULL,                           /* req */
    5104             :                 NULL,                           /* dirfsp */
    5105             :                 smb_fname,                      /* fname */
    5106             :                 FILE_GENERIC_READ,              /* access_mask */
    5107             :                 FILE_SHARE_READ | FILE_SHARE_WRITE, /* share_access */
    5108             :                 FILE_OPEN,                      /* create_disposition */
    5109             :                 0,                              /* create_options */
    5110             :                 0,                              /* file_attributes */
    5111             :                 INTERNAL_OPEN_ONLY,             /* oplock_request */
    5112             :                 NULL,                           /* lease */
    5113             :                 0,                              /* allocation_size */
    5114             :                 0,                              /* private_flags */
    5115             :                 NULL,                           /* sd */
    5116             :                 NULL,                           /* ea_list */
    5117             :                 &fsp,                               /* result */
    5118             :                 NULL,                           /* psbuf */
    5119             :                 NULL, NULL);                    /* create context */
    5120           2 :         if (!NT_STATUS_IS_OK(status)) {
    5121           0 :                 DBG_INFO("Opening [%s] failed [%s]\n",
    5122             :                          smb_fname_str_dbg(smb_fname), nt_errstr(status));
    5123           0 :                 ok = false;
    5124           0 :                 goto out;
    5125             :         }
    5126             : 
    5127           2 :         file_data = talloc_zero_array(talloc_tos(),
    5128             :                                       uint8_t,
    5129             :                                       plist_file_size + 1);
    5130           2 :         if (file_data == NULL) {
    5131           0 :                 ok = false;
    5132           0 :                 goto out;
    5133             :         }
    5134             : 
    5135           2 :         nread = SMB_VFS_NEXT_PREAD(handle, fsp, file_data, plist_file_size, 0);
    5136           2 :         if (nread != plist_file_size) {
    5137           0 :                 DBG_ERR("Short read on [%s]: %zu/%zd\n",
    5138             :                         fsp_str_dbg(fsp), nread, plist_file_size);
    5139           0 :                 ok = false;
    5140           0 :                 goto out;
    5141             : 
    5142             :         }
    5143             : 
    5144           2 :         status = close_file_free(NULL, &fsp, NORMAL_CLOSE);
    5145           2 :         if (!NT_STATUS_IS_OK(status)) {
    5146           0 :                 DBG_ERR("close_file failed: %s\n", nt_errstr(status));
    5147           0 :                 ok = false;
    5148           0 :                 goto out;
    5149             :         }
    5150             : 
    5151           2 :         lines = file_lines_parse((char *)file_data,
    5152             :                                  plist_file_size,
    5153             :                                  &numlines,
    5154             :                                  talloc_tos());
    5155           2 :         if (lines == NULL) {
    5156           0 :                 ok = false;
    5157           0 :                 goto out;
    5158             :         }
    5159             : 
    5160           2 :         band_size_line = fruit_get_bandsize_line(lines, numlines);
    5161           2 :         if (band_size_line == NULL) {
    5162           0 :                 DBG_ERR("Didn't find band-size key in [%s]\n",
    5163             :                         smb_fname_str_dbg(smb_fname));
    5164           0 :                 ok = false;
    5165           0 :                 goto out;
    5166             :         }
    5167             : 
    5168           2 :         ok = fruit_get_bandsize_from_line(band_size_line, band_size);
    5169           2 :         if (!ok) {
    5170           0 :                 DBG_ERR("fruit_get_bandsize_from_line failed\n");
    5171           0 :                 goto out;
    5172             :         }
    5173             : 
    5174           2 :         DBG_DEBUG("Parsed band-size [%zu] for [%s]\n", *band_size, plist);
    5175             : 
    5176           2 : out:
    5177           2 :         if (fsp != NULL) {
    5178           0 :                 status = close_file_free(NULL, &fsp, NORMAL_CLOSE);
    5179           0 :                 if (!NT_STATUS_IS_OK(status)) {
    5180           0 :                         DBG_ERR("close_file failed: %s\n", nt_errstr(status));
    5181             :                 }
    5182             :         }
    5183           2 :         TALLOC_FREE(plist);
    5184           2 :         TALLOC_FREE(smb_fname);
    5185           2 :         TALLOC_FREE(file_data);
    5186           2 :         TALLOC_FREE(lines);
    5187           2 :         return ok;
    5188             : }
    5189             : 
    5190             : struct fruit_disk_free_state {
    5191             :         off_t total_size;
    5192             : };
    5193             : 
    5194           2 : static bool fruit_get_num_bands(vfs_handle_struct *handle,
    5195             :                                 const char *bundle,
    5196             :                                 size_t *_nbands)
    5197             : {
    5198           2 :         char *path = NULL;
    5199           2 :         struct smb_filename *bands_dir = NULL;
    5200           2 :         struct smb_Dir *dir_hnd = NULL;
    5201           2 :         const char *dname = NULL;
    5202           2 :         char *talloced = NULL;
    5203             :         size_t nbands;
    5204             :         NTSTATUS status;
    5205             : 
    5206           2 :         path = talloc_asprintf(talloc_tos(),
    5207             :                                "%s/%s/bands",
    5208           2 :                                handle->conn->connectpath,
    5209             :                                bundle);
    5210           2 :         if (path == NULL) {
    5211           0 :                 return false;
    5212             :         }
    5213             : 
    5214           2 :         bands_dir = synthetic_smb_fname(talloc_tos(),
    5215             :                                         path,
    5216             :                                         NULL,
    5217             :                                         NULL,
    5218             :                                         0,
    5219             :                                         0);
    5220           2 :         TALLOC_FREE(path);
    5221           2 :         if (bands_dir == NULL) {
    5222           0 :                 return false;
    5223             :         }
    5224             : 
    5225           2 :         status = OpenDir(talloc_tos(),
    5226           2 :                          handle->conn,
    5227             :                          bands_dir,
    5228             :                          NULL,
    5229             :                          0,
    5230             :                          &dir_hnd);
    5231           2 :         if (!NT_STATUS_IS_OK(status)) {
    5232           0 :                 TALLOC_FREE(bands_dir);
    5233           0 :                 errno = map_errno_from_nt_status(status);
    5234           0 :                 return false;
    5235             :         }
    5236             : 
    5237           2 :         nbands = 0;
    5238             : 
    5239          10 :         while ((dname = ReadDirName(dir_hnd, &talloced)) != NULL) {
    5240           8 :                 if (ISDOT(dname) || ISDOTDOT(dname)) {
    5241           4 :                         continue;
    5242             :                 }
    5243           4 :                 nbands++;
    5244             :         }
    5245           2 :         TALLOC_FREE(dir_hnd);
    5246             : 
    5247           2 :         DBG_DEBUG("%zu bands in [%s]\n", nbands, smb_fname_str_dbg(bands_dir));
    5248             : 
    5249           2 :         TALLOC_FREE(bands_dir);
    5250             : 
    5251           2 :         *_nbands = nbands;
    5252           2 :         return true;
    5253             : }
    5254             : 
    5255         184 : static bool fruit_tmsize_do_dirent(vfs_handle_struct *handle,
    5256             :                                    struct fruit_disk_free_state *state,
    5257             :                                    const char *name)
    5258             : {
    5259             :         bool ok;
    5260         184 :         char *p = NULL;
    5261         184 :         size_t sparsebundle_strlen = strlen("sparsebundle");
    5262         184 :         size_t bandsize = 0;
    5263             :         size_t nbands;
    5264             :         off_t tm_size;
    5265             : 
    5266         184 :         p = strstr(name, "sparsebundle");
    5267         184 :         if (p == NULL) {
    5268         182 :                 return true;
    5269             :         }
    5270             : 
    5271           2 :         if (p[sparsebundle_strlen] != '\0') {
    5272           0 :                 return true;
    5273             :         }
    5274             : 
    5275           2 :         DBG_DEBUG("Processing sparsebundle [%s]\n", name);
    5276             : 
    5277           2 :         ok = fruit_get_bandsize(handle, name, &bandsize);
    5278           2 :         if (!ok) {
    5279             :                 /*
    5280             :                  * Beware of race conditions: this may be an uninitialized
    5281             :                  * Info.plist that a client is just creating. We don't want let
    5282             :                  * this to trigger complete failure.
    5283             :                  */
    5284           0 :                 DBG_ERR("Processing sparsebundle [%s] failed\n", name);
    5285           0 :                 return true;
    5286             :         }
    5287             : 
    5288           2 :         ok = fruit_get_num_bands(handle, name, &nbands);
    5289           2 :         if (!ok) {
    5290             :                 /*
    5291             :                  * Beware of race conditions: this may be a backup sparsebundle
    5292             :                  * in an early stage lacking a bands subdirectory. We don't want
    5293             :                  * let this to trigger complete failure.
    5294             :                  */
    5295           0 :                 DBG_ERR("Processing sparsebundle [%s] failed\n", name);
    5296           0 :                 return true;
    5297             :         }
    5298             : 
    5299             :         /*
    5300             :          * Arithmetic on 32-bit systems may cause overflow, depending on
    5301             :          * size_t precision. First we check its unlikely, then we
    5302             :          * force the precision into target off_t, then we check that
    5303             :          * the total did not overflow either.
    5304             :          */
    5305           2 :         if (bandsize > SIZE_MAX/nbands) {
    5306           0 :                 DBG_ERR("tmsize potential overflow: bandsize [%zu] nbands [%zu]\n",
    5307             :                         bandsize, nbands);
    5308           0 :                 return false;
    5309             :         }
    5310           2 :         tm_size = (off_t)bandsize * (off_t)nbands;
    5311             : 
    5312           2 :         if (state->total_size + tm_size < state->total_size) {
    5313           0 :                 DBG_ERR("tm total size overflow: bandsize [%zu] nbands [%zu]\n",
    5314             :                         bandsize, nbands);
    5315           0 :                 return false;
    5316             :         }
    5317             : 
    5318           2 :         state->total_size += tm_size;
    5319             : 
    5320           2 :         DBG_DEBUG("[%s] tm_size [%jd] total_size [%jd]\n",
    5321             :                   name, (intmax_t)tm_size, (intmax_t)state->total_size);
    5322             : 
    5323           2 :         return true;
    5324             : }
    5325             : 
    5326             : /**
    5327             :  * Calculate used size of a TimeMachine volume
    5328             :  *
    5329             :  * This assumes that the volume is used only for TimeMachine.
    5330             :  *
    5331             :  * - readdir(basedir of share), then
    5332             :  * - for every element that matches regex "^\(.*\)\.sparsebundle$" :
    5333             :  * - parse "\1.sparsebundle/Info.plist" and read the band-size XML key
    5334             :  * - count band files in "\1.sparsebundle/bands/"
    5335             :  * - calculate used size of all bands: band_count * band_size
    5336             :  **/
    5337           2 : static uint64_t fruit_disk_free(vfs_handle_struct *handle,
    5338             :                                 const struct smb_filename *smb_fname,
    5339             :                                 uint64_t *_bsize,
    5340             :                                 uint64_t *_dfree,
    5341             :                                 uint64_t *_dsize)
    5342             : {
    5343           2 :         struct fruit_config_data *config = NULL;
    5344           2 :         struct fruit_disk_free_state state = {0};
    5345           2 :         struct smb_Dir *dir_hnd = NULL;
    5346           2 :         const char *dname = NULL;
    5347           2 :         char *talloced = NULL;
    5348             :         uint64_t dfree;
    5349             :         uint64_t dsize;
    5350             :         bool ok;
    5351             :         NTSTATUS status;
    5352             : 
    5353           2 :         SMB_VFS_HANDLE_GET_DATA(handle, config,
    5354             :                                 struct fruit_config_data,
    5355             :                                 return UINT64_MAX);
    5356             : 
    5357           2 :         if (!config->time_machine ||
    5358           2 :             config->time_machine_max_size == 0)
    5359             :         {
    5360           0 :                 return SMB_VFS_NEXT_DISK_FREE(handle,
    5361             :                                               smb_fname,
    5362             :                                               _bsize,
    5363             :                                               _dfree,
    5364             :                                               _dsize);
    5365             :         }
    5366             : 
    5367           2 :         status = OpenDir(talloc_tos(),
    5368           2 :                          handle->conn,
    5369             :                          smb_fname,
    5370             :                          NULL,
    5371             :                          0,
    5372             :                          &dir_hnd);
    5373           2 :         if (!NT_STATUS_IS_OK(status)) {
    5374           0 :                 errno = map_errno_from_nt_status(status);
    5375           0 :                 return UINT64_MAX;
    5376             :         }
    5377             : 
    5378         186 :         while ((dname = ReadDirName(dir_hnd, &talloced)) != NULL) {
    5379         184 :                 ok = fruit_tmsize_do_dirent(handle, &state, dname);
    5380         184 :                 if (!ok) {
    5381           0 :                         TALLOC_FREE(talloced);
    5382           0 :                         TALLOC_FREE(dir_hnd);
    5383           0 :                         return UINT64_MAX;
    5384             :                 }
    5385         184 :                 TALLOC_FREE(talloced);
    5386             :         }
    5387             : 
    5388           2 :         TALLOC_FREE(dir_hnd);
    5389             : 
    5390           2 :         dsize = config->time_machine_max_size / 512;
    5391           2 :         dfree = dsize - (state.total_size / 512);
    5392           2 :         if (dfree > dsize) {
    5393           0 :                 dfree = 0;
    5394             :         }
    5395             : 
    5396           2 :         *_bsize = 512;
    5397           2 :         *_dsize = dsize;
    5398           2 :         *_dfree = dfree;
    5399           2 :         return dfree / 2;
    5400             : }
    5401             : 
    5402        3828 : static uint64_t fruit_fs_file_id(struct vfs_handle_struct *handle,
    5403             :                                  const SMB_STRUCT_STAT *psbuf)
    5404             : {
    5405        3828 :         struct fruit_config_data *config = NULL;
    5406             : 
    5407        3828 :         SMB_VFS_HANDLE_GET_DATA(handle, config,
    5408             :                                 struct fruit_config_data,
    5409             :                                 return 0);
    5410             : 
    5411        3828 :         if (global_fruit_config.nego_aapl &&
    5412         870 :             config->aapl_zero_file_id)
    5413             :         {
    5414         870 :                 return 0;
    5415             :         }
    5416             : 
    5417        2958 :         return SMB_VFS_NEXT_FS_FILE_ID(handle, psbuf);
    5418             : }
    5419             : 
    5420             : static struct vfs_fn_pointers vfs_fruit_fns = {
    5421             :         .connect_fn = fruit_connect,
    5422             :         .disk_free_fn = fruit_disk_free,
    5423             : 
    5424             :         /* File operations */
    5425             :         .fchmod_fn = fruit_fchmod,
    5426             :         .unlinkat_fn = fruit_unlinkat,
    5427             :         .renameat_fn = fruit_renameat,
    5428             :         .openat_fn = fruit_openat,
    5429             :         .close_fn = fruit_close,
    5430             :         .pread_fn = fruit_pread,
    5431             :         .pwrite_fn = fruit_pwrite,
    5432             :         .pread_send_fn = fruit_pread_send,
    5433             :         .pread_recv_fn = fruit_pread_recv,
    5434             :         .pwrite_send_fn = fruit_pwrite_send,
    5435             :         .pwrite_recv_fn = fruit_pwrite_recv,
    5436             :         .fsync_send_fn = fruit_fsync_send,
    5437             :         .fsync_recv_fn = fruit_fsync_recv,
    5438             :         .stat_fn = fruit_stat,
    5439             :         .lstat_fn = fruit_lstat,
    5440             :         .fstat_fn = fruit_fstat,
    5441             :         .fstreaminfo_fn = fruit_fstreaminfo,
    5442             :         .fntimes_fn = fruit_fntimes,
    5443             :         .ftruncate_fn = fruit_ftruncate,
    5444             :         .fallocate_fn = fruit_fallocate,
    5445             :         .create_file_fn = fruit_create_file,
    5446             :         .freaddir_attr_fn = fruit_freaddir_attr,
    5447             :         .offload_read_send_fn = fruit_offload_read_send,
    5448             :         .offload_read_recv_fn = fruit_offload_read_recv,
    5449             :         .offload_write_send_fn = fruit_offload_write_send,
    5450             :         .offload_write_recv_fn = fruit_offload_write_recv,
    5451             :         .fs_file_id_fn = fruit_fs_file_id,
    5452             : 
    5453             :         /* NT ACL operations */
    5454             :         .fget_nt_acl_fn = fruit_fget_nt_acl,
    5455             :         .fset_nt_acl_fn = fruit_fset_nt_acl,
    5456             : };
    5457             : 
    5458             : static_decl_vfs;
    5459         339 : NTSTATUS vfs_fruit_init(TALLOC_CTX *ctx)
    5460             : {
    5461         339 :         NTSTATUS ret = smb_register_vfs(SMB_VFS_INTERFACE_VERSION, "fruit",
    5462             :                                         &vfs_fruit_fns);
    5463         339 :         if (!NT_STATUS_IS_OK(ret)) {
    5464           0 :                 return ret;
    5465             :         }
    5466             : 
    5467         339 :         vfs_fruit_debug_level = debug_add_class("fruit");
    5468         339 :         if (vfs_fruit_debug_level == -1) {
    5469           0 :                 vfs_fruit_debug_level = DBGC_VFS;
    5470           0 :                 DEBUG(0, ("%s: Couldn't register custom debugging class!\n",
    5471             :                           "vfs_fruit_init"));
    5472             :         } else {
    5473         339 :                 DEBUG(10, ("%s: Debug class number of '%s': %d\n",
    5474             :                            "vfs_fruit_init","fruit",vfs_fruit_debug_level));
    5475             :         }
    5476             : 
    5477         339 :         return ret;
    5478             : }

Generated by: LCOV version 1.14