Diff for /embedaddon/rsync/acls.c between versions 1.1 and 1.1.1.3

version 1.1, 2012/02/17 15:09:30 version 1.1.1.3, 2016/11/01 09:54:32
Line 3 Line 3
  *   *
  * Copyright (C) 1996 Andrew Tridgell   * Copyright (C) 1996 Andrew Tridgell
  * Copyright (C) 1996 Paul Mackerras   * Copyright (C) 1996 Paul Mackerras
 * Copyright (C) 2006-2009 Wayne Davison * Copyright (C) 2006-2015 Wayne Davison
  *   *
  * This program is free software; you can redistribute it and/or modify   * This program is free software; you can redistribute it and/or modify
  * it under the terms of the GNU General Public License as published by   * it under the terms of the GNU General Public License as published by
Line 78  typedef struct rsync_acl { Line 78  typedef struct rsync_acl {
         uchar other_obj;          uchar other_obj;
 } rsync_acl;  } rsync_acl;
   
   typedef struct nfs4_acl {
           char *nfs4_acl_text;
           ssize_t nfs4_acl_len;
   } nfs4_acl;
   
 typedef struct {  typedef struct {
         rsync_acl racl;          rsync_acl racl;
         SMB_ACL_T sacl;          SMB_ACL_T sacl;
 } acl_duo;  } acl_duo;
   
   typedef struct {
           nfs4_acl nacl;
           SMB_ACL_T sacl;
   } nfs4_duo;
   
 static const rsync_acl empty_rsync_acl = {  static const rsync_acl empty_rsync_acl = {
         {NULL, 0}, NO_ENTRY, NO_ENTRY, NO_ENTRY, NO_ENTRY          {NULL, 0}, NO_ENTRY, NO_ENTRY, NO_ENTRY, NO_ENTRY
 };  };
   static const nfs4_acl empty_nfs4_acl = {
           NULL, -1
   };
   
 static item_list access_acl_list = EMPTY_ITEM_LIST;  static item_list access_acl_list = EMPTY_ITEM_LIST;
 static item_list default_acl_list = EMPTY_ITEM_LIST;  static item_list default_acl_list = EMPTY_ITEM_LIST;
   static item_list nfs4_acl_list = EMPTY_ITEM_LIST;
   
 static size_t prior_access_count = (size_t)-1;  static size_t prior_access_count = (size_t)-1;
 static size_t prior_default_count = (size_t)-1;  static size_t prior_default_count = (size_t)-1;
   static size_t prior_nfs4_count = (size_t)-1;
   
 /* === Calculations on ACL types === */  /* === Calculations on ACL types === */
   
Line 175  static rsync_acl *create_racl(void) Line 190  static rsync_acl *create_racl(void)
         return racl;          return racl;
 }  }
   
   static nfs4_acl *create_nfs4_acl(void)
   {
           nfs4_acl *nacl = new(nfs4_acl);
   
           if (!nacl)
                   out_of_memory("create_nfs4_acl");
           *nacl = empty_nfs4_acl;
   
           return nacl;
   }
   
 static BOOL ida_entries_equal(const ida_entries *ial1, const ida_entries *ial2)  static BOOL ida_entries_equal(const ida_entries *ial1, const ida_entries *ial2)
 {  {
         id_access *ida1, *ida2;          id_access *ida1, *ida2;
Line 199  static BOOL rsync_acl_equal(const rsync_acl *racl1, co Line 225  static BOOL rsync_acl_equal(const rsync_acl *racl1, co
             && ida_entries_equal(&racl1->names, &racl2->names);              && ida_entries_equal(&racl1->names, &racl2->names);
 }  }
   
   static BOOL nfs4_acl_equal(const nfs4_acl *nacl1, const nfs4_acl *nacl2)
   {
           return (strcmp(nacl1->nfs4_acl_text, nacl2->nfs4_acl_text) == 0);
   }
   
 /* Are the extended (non-permission-bit) entries equal?  If so, the rest of  /* Are the extended (non-permission-bit) entries equal?  If so, the rest of
  * the ACL will be handled by the normal mode-preservation code.  This is   * the ACL will be handled by the normal mode-preservation code.  This is
  * only meaningful for access ACLs!  Note: the 1st arg is a fully-populated   * only meaningful for access ACLs!  Note: the 1st arg is a fully-populated
Line 232  static void rsync_acl_free(rsync_acl *racl) Line 263  static void rsync_acl_free(rsync_acl *racl)
         *racl = empty_rsync_acl;          *racl = empty_rsync_acl;
 }  }
   
   static void nfs4_acl_free(nfs4_acl *nacl)
   {
           if (nacl->nfs4_acl_text)
                   free(nacl->nfs4_acl_text);
           *nacl = empty_nfs4_acl;
   }
   
 void free_acl(stat_x *sxp)  void free_acl(stat_x *sxp)
 {  {
         if (sxp->acc_acl) {          if (sxp->acc_acl) {
Line 244  void free_acl(stat_x *sxp) Line 282  void free_acl(stat_x *sxp)
                 free(sxp->def_acl);                  free(sxp->def_acl);
                 sxp->def_acl = NULL;                  sxp->def_acl = NULL;
         }          }
           if (sxp->nfs4_acl) {
                   nfs4_acl_free(sxp->nfs4_acl);
                   free(sxp->nfs4_acl);
                   sxp->nfs4_acl = NULL;
           }
 }  }
   
 #ifdef SMB_ACL_NEED_SORT  #ifdef SMB_ACL_NEED_SORT
Line 423  static BOOL pack_smb_acl(SMB_ACL_T *smb_acl, const rsy Line 466  static BOOL pack_smb_acl(SMB_ACL_T *smb_acl, const rsy
 #ifdef ACLS_NEED_MASK  #ifdef ACLS_NEED_MASK
         mask_bits = racl->mask_obj == NO_ENTRY ? racl->group_obj & ~NO_ENTRY : racl->mask_obj;          mask_bits = racl->mask_obj == NO_ENTRY ? racl->group_obj & ~NO_ENTRY : racl->mask_obj;
         COE( sys_acl_create_entry,(smb_acl, &entry) );          COE( sys_acl_create_entry,(smb_acl, &entry) );
        COE( sys_acl_set_info,(entry, SMB_ACL_MASK, mask_bits, NULL) );        COE( sys_acl_set_info,(entry, SMB_ACL_MASK, mask_bits, 0) );
 #else  #else
         if (racl->mask_obj != NO_ENTRY) {          if (racl->mask_obj != NO_ENTRY) {
                 COE( sys_acl_create_entry,(smb_acl, &entry) );                  COE( sys_acl_create_entry,(smb_acl, &entry) );
Line 474  static int find_matching_rsync_acl(const rsync_acl *ra Line 517  static int find_matching_rsync_acl(const rsync_acl *ra
         return *match;          return *match;
 }  }
   
   static int find_matching_nfs4_acl(const nfs4_acl *nacl, const item_list *nfs4_acl_list)
   {
           static int nfs4_match = -1;
           int *match = &nfs4_match;
           size_t count = nfs4_acl_list->count;
   
           if (*match == -1)
                   *match = nfs4_acl_list->count - 1;
           while (count--) {
                   nfs4_acl *base = nfs4_acl_list->items;
                   if (nfs4_acl_equal(base + *match, nacl))
                           return *match;
                   if (!(*match)--)
                           *match = nfs4_acl_list->count - 1;
           }
   
           *match = -1;
           return *match;
   }
   
 static int get_rsync_acl(const char *fname, rsync_acl *racl,  static int get_rsync_acl(const char *fname, rsync_acl *racl,
                          SMB_ACL_TYPE_T type, mode_t mode)                           SMB_ACL_TYPE_T type, mode_t mode)
 {  {
Line 544  static int get_rsync_acl(const char *fname, rsync_acl  Line 607  static int get_rsync_acl(const char *fname, rsync_acl 
 /* Return the Access Control List for the given filename. */  /* Return the Access Control List for the given filename. */
 int get_acl(const char *fname, stat_x *sxp)  int get_acl(const char *fname, stat_x *sxp)
 {  {
           if (sys_acl_get_brand_file(fname, &sxp->brand) < 0)
                   return -1;
   
           if (sxp->brand == SMB_ACL_BRAND_NFS4) {
                   SMB_ACL_T sacl;
                   if ((sacl = sys_acl_get_file(fname, SMB_ACL_TYPE_NFS4)) == NULL)
                           return -1;
   
                   sxp->nfs4_acl = create_nfs4_acl();
                   sxp->nfs4_acl->nfs4_acl_text = acl_to_text(sacl, &sxp->nfs4_acl->nfs4_acl_len);
   
                   sys_acl_free_acl(sacl);
                   return 0;
           }
   
         sxp->acc_acl = create_racl();          sxp->acc_acl = create_racl();
   
         if (S_ISREG(sxp->st.st_mode) || S_ISDIR(sxp->st.st_mode)) {          if (S_ISREG(sxp->st.st_mode) || S_ISDIR(sxp->st.st_mode)) {
Line 560  int get_acl(const char *fname, stat_x *sxp) Line 638  int get_acl(const char *fname, stat_x *sxp)
                 if (!preserve_devices)                  if (!preserve_devices)
 #endif  #endif
                         return 0;                          return 0;
        }        } else if (IS_MISSING_FILE(sxp->st))
                 return 0;
   
         if (get_rsync_acl(fname, sxp->acc_acl, SMB_ACL_TYPE_ACCESS,          if (get_rsync_acl(fname, sxp->acc_acl, SMB_ACL_TYPE_ACCESS,
                           sxp->st.st_mode) < 0) {                            sxp->st.st_mode) < 0) {
Line 651  static void send_rsync_acl(int f, rsync_acl *racl, SMB Line 730  static void send_rsync_acl(int f, rsync_acl *racl, SMB
         }          }
 }  }
   
   static void send_nfs4_acl(int f, nfs4_acl *nacl, item_list *nfs4_list)
   {
           int ndx = find_matching_nfs4_acl(nacl, nfs4_list);
   
           /* Send 0 (-1 + 1) to indicate that literal ACL data follows. */
           write_varint(f, ndx + 1);
   
           if (ndx < 0) {
                   nfs4_acl *new_nacl = EXPAND_ITEM_LIST(&nfs4_acl_list, nfs4_acl, 1000);
   
                   write_varint(f, nacl->nfs4_acl_len);
                   write_buf(f, nacl->nfs4_acl_text, nacl->nfs4_acl_len);
   
                   *new_nacl = *nacl;
                   *nacl = empty_nfs4_acl;
           }
   }
   
   
 /* Send the ACL from the stat_x structure down the indicated file descriptor.  /* Send the ACL from the stat_x structure down the indicated file descriptor.
  * This also frees the ACL data. */   * This also frees the ACL data. */
 void send_acl(int f, stat_x *sxp)  void send_acl(int f, stat_x *sxp)
Line 662  void send_acl(int f, stat_x *sxp) Line 760  void send_acl(int f, stat_x *sxp)
         /* Avoid sending values that can be inferred from other data. */          /* Avoid sending values that can be inferred from other data. */
         rsync_acl_strip_perms(sxp);          rsync_acl_strip_perms(sxp);
   
           write_varint(f, SMB_ACL_TYPE_ACCESS);
         send_rsync_acl(f, sxp->acc_acl, SMB_ACL_TYPE_ACCESS, &access_acl_list);          send_rsync_acl(f, sxp->acc_acl, SMB_ACL_TYPE_ACCESS, &access_acl_list);
   
         if (S_ISDIR(sxp->st.st_mode)) {          if (S_ISDIR(sxp->st.st_mode)) {
                 if (!sxp->def_acl)                  if (!sxp->def_acl)
                         sxp->def_acl = create_racl();                          sxp->def_acl = create_racl();
   
                   write_varint(f, SMB_ACL_TYPE_DEFAULT);
                 send_rsync_acl(f, sxp->def_acl, SMB_ACL_TYPE_DEFAULT, &default_acl_list);                  send_rsync_acl(f, sxp->def_acl, SMB_ACL_TYPE_DEFAULT, &default_acl_list);
         }          }
 }  }
Line 817  static int cache_rsync_acl(rsync_acl *racl, SMB_ACL_TY Line 917  static int cache_rsync_acl(rsync_acl *racl, SMB_ACL_TY
         return ndx;          return ndx;
 }  }
   
   static int cache_nfs4_acl(nfs4_acl *nacl, item_list *nfs4_list)
   {
           int ndx;
   
           if (!nacl)
                   ndx = -1;
           else if ((ndx = find_matching_nfs4_acl(nacl, nfs4_list)) == -1) {
                   nfs4_duo *new_duo;
                   ndx = nfs4_list->count;
                   new_duo = EXPAND_ITEM_LIST(nfs4_list, nfs4_duo, 1000);
                   new_duo->nacl = *nacl;
                   new_duo->sacl = NULL;
                   *nacl = empty_nfs4_acl;
           }
   
           return ndx;
   }
   
   
 /* Turn the ACL data in stat_x into cached ACL data, setting the index  /* Turn the ACL data in stat_x into cached ACL data, setting the index
  * values in the file struct. */   * values in the file struct. */
 void cache_tmp_acl(struct file_struct *file, stat_x *sxp)  void cache_tmp_acl(struct file_struct *file, stat_x *sxp)
 {  {
           if (sxp->brand == SMB_ACL_BRAND_NFS4) {
                   if (prior_nfs4_count == (size_t)-1)
                           prior_nfs4_count = nfs4_acl_list.count;
   
                   F_ACL(file) = cache_nfs4_acl(sxp->nfs4_acl, &nfs4_acl_list);
                   return;
           }
   
         if (prior_access_count == (size_t)-1)          if (prior_access_count == (size_t)-1)
                 prior_access_count = access_acl_list.count;                  prior_access_count = access_acl_list.count;
   
Line 850  static void uncache_duo_acls(item_list *duo_list, size Line 977  static void uncache_duo_acls(item_list *duo_list, size
         }          }
 }  }
   
   static void uncache_nfs4_acls(item_list *nfs4_list, size_t start)
   {
           nfs4_duo *nfs4_item = nfs4_list->items;
           nfs4_duo *nfs4_start = nfs4_item + start;
   
           nfs4_item += nfs4_list->count;
           nfs4_list->count = start;
   
           while (nfs4_item-- > nfs4_start) {
                   nfs4_acl_free(&nfs4_item->nacl);
                   if (nfs4_item->sacl)
                           sys_acl_free_acl(nfs4_item->sacl);
           }
   }
   
 void uncache_tmp_acls(void)  void uncache_tmp_acls(void)
 {  {
         if (prior_access_count != (size_t)-1) {          if (prior_access_count != (size_t)-1) {
Line 861  void uncache_tmp_acls(void) Line 1003  void uncache_tmp_acls(void)
                 uncache_duo_acls(&default_acl_list, prior_default_count);                  uncache_duo_acls(&default_acl_list, prior_default_count);
                 prior_default_count = (size_t)-1;                  prior_default_count = (size_t)-1;
         }          }
           if (prior_nfs4_count != (size_t)-1) {
                   uncache_nfs4_acls(&nfs4_acl_list, prior_nfs4_count);
                   prior_nfs4_count = (size_t)-1;
           }
 }  }
   
 #ifndef HAVE_OSX_ACLS  #ifndef HAVE_OSX_ACLS
Line 1013  static int set_rsync_acl(const char *fname, acl_duo *d Line 1159  static int set_rsync_acl(const char *fname, acl_duo *d
         return 0;          return 0;
 }  }
   
   
 /* Given a fname, this sets extended access ACL entries, the default ACL (for a  /* Given a fname, this sets extended access ACL entries, the default ACL (for a
  * dir), and the regular mode bits on the file.  Call this with fname set to   * dir), and the regular mode bits on the file.  Call this with fname set to
  * NULL to just check if the ACL is different.   * NULL to just check if the ACL is different.
Line 1032  int set_acl(const char *fname, const struct file_struc Line 1179  int set_acl(const char *fname, const struct file_struc
                 return -1;                  return -1;
         }          }
   
           if (sxp->brand == SMB_ACL_BRAND_NFS4) {
                   ndx = F_ACL(file);
                   if (ndx >= 0 && (size_t)ndx < nfs4_acl_list.count) {
                           nfs4_duo *duo_item = nfs4_acl_list.items;
                           duo_item += ndx;
                           changed = 1;
   
                           if (!duo_item->sacl) {
                                   duo_item->sacl = acl_from_text(duo_item->nacl.nfs4_acl_text);
                                   if (!duo_item->sacl)
                                           return -1;
                           }
   
                           if (!dry_run && fname) {
                                   if (sys_acl_set_file(fname, SMB_ACL_TYPE_NFS4, duo_item->sacl) < 0) {
                                           rsyserr(FERROR_XFER, errno, "set_acl: sys_acl_set_file(%s, %s)",
                                                   fname, str_acl_type(SMB_ACL_TYPE_NFS4));
                                           return -1;
                                   }
   
                                   return changed;
                           }
                   }
           }
   
   
         ndx = F_ACL(file);          ndx = F_ACL(file);
         if (ndx >= 0 && (size_t)ndx < access_acl_list.count) {          if (ndx >= 0 && (size_t)ndx < access_acl_list.count) {
                 acl_duo *duo_item = access_acl_list.items;                  acl_duo *duo_item = access_acl_list.items;
Line 1143  int default_perms_for_dir(const char *dir) Line 1316  int default_perms_for_dir(const char *dir)
         /* Apply the permission-bit entries of the default ACL, if any. */          /* Apply the permission-bit entries of the default ACL, if any. */
         if (racl.user_obj != NO_ENTRY) {          if (racl.user_obj != NO_ENTRY) {
                 perms = rsync_acl_get_perms(&racl);                  perms = rsync_acl_get_perms(&racl);
                if (verbose > 2)                if (DEBUG_GTE(ACL, 1))
                         rprintf(FINFO, "got ACL-based default perms %o for directory %s\n", perms, dir);                          rprintf(FINFO, "got ACL-based default perms %o for directory %s\n", perms, dir);
         }          }
   

Removed from v.1.1  
changed lines
  Added in v.1.1.1.3


FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>