[SCM] The rsync repository. - branch master updated

Rsync CVS commit messages rsync-cvs at lists.samba.org
Tue Feb 22 11:32:33 MST 2011


The branch, master has been updated
       via  eee2c77 Some uid/gid fixes for (id_t)-1 and other large ID values.
      from  7766e67 Allow a failure of EINVAL to mean no ACLs are available. (If our POSIX types aren't valid, we can't handle the ACLs.)

;a=shortlog;h=master


- Log -----------------------------------------------------------------
commit eee2c77a93d466c11b6162054987114b9a342d8c
Author: Wayne Davison <wayned at samba.org>
Date:   Tue Feb 22 09:31:54 2011 -0800

    Some uid/gid fixes for (id_t)-1 and other large ID values.
    
    The code now avoids any special internal meaning for uid/gid -1, which
    allows it to be mapped to a better value (use 4294967295 instead of -1
    as the ID to map).  Replaced atol() with something than can return a
    value > 0x7FFFFFFF and that will error-out if the value overflows.  If
    chown() is called with a uid or gid of -1, complain that the ID is not
    settable and signal a transfer error.  Fixes bug 6936.

-----------------------------------------------------------------------

Summary of changes:
 rsync.c   |   10 +++++++---
 uidlist.c |   47 ++++++++++++++++++++++++++++++++---------------
 2 files changed, 39 insertions(+), 18 deletions(-)


Changeset truncated at 500 lines:

diff --git a/rsync.c b/rsync.c
index 24b9c75..c42d553 100644
--- a/rsync.c
+++ b/rsync.c
@@ -527,9 +527,9 @@ int set_file_attrs(const char *fname, struct file_struct *file, stat_x *sxp,
 			}
 		}
 		if (am_root >= 0) {
-			if (do_lchown(fname,
-			    change_uid ? (uid_t)F_OWNER(file) : sxp->st.st_uid,
-			    change_gid ? (gid_t)F_GROUP(file) : sxp->st.st_gid) != 0) {
+			uid_t uid = change_uid ? (uid_t)F_OWNER(file) : sxp->st.st_uid;
+			gid_t gid = change_gid ? (gid_t)F_GROUP(file) : sxp->st.st_gid;
+			if (do_lchown(fname, uid, gid) != 0) {
 				/* We shouldn't have attempted to change uid
 				 * or gid unless have the privilege. */
 				rsyserr(FERROR_XFER, errno, "%s %s failed",
@@ -537,6 +537,10 @@ int set_file_attrs(const char *fname, struct file_struct *file, stat_x *sxp,
 				    full_fname(fname));
 				goto cleanup;
 			}
+			if (uid == (uid_t)-1 && sxp->st.st_uid != (uid_t)-1)
+				rprintf(FERROR_XFER, "uid 4294967295 (-1) is impossible to set on %s\n", full_fname(fname));
+			if (gid == (gid_t)-1 && sxp->st.st_gid != (gid_t)-1)
+				rprintf(FERROR_XFER, "gid 4294967295 (-1) is impossible to set on %s\n", full_fname(fname));
 			/* A lchown had been done, so we need to re-stat if
 			 * the destination had the setuid or setgid bits set
 			 * (due to the side effect of the chown call). */
diff --git a/uidlist.c b/uidlist.c
index f771fb6..c6639ad 100644
--- a/uidlist.c
+++ b/uidlist.c
@@ -43,8 +43,6 @@ extern char *groupmap;
 # endif
 #endif
 
-#define GID_NONE ((gid_t)-1)
-
 #define NFLAGS_WILD_NAME_MATCH (1<<0)
 #define NFLAGS_NAME_MATCH (1<<1)
 
@@ -58,6 +56,26 @@ struct idlist {
 static struct idlist *uidlist, *uidmap;
 static struct idlist *gidlist, *gidmap;
 
+static id_t id_parse(const char *num_str)
+{
+	id_t tmp, num = 0;
+	const char *cp = num_str;
+
+	while (*cp) {
+		if (!isDigit(cp)) {
+		  invalid_num:
+			rprintf(FERROR, "Invalid ID number: %s\n", num_str);
+			exit_cleanup(RERR_SYNTAX);
+		}
+		tmp = num * 10 + *cp++ - '0';
+		if (tmp < num)
+			goto invalid_num;
+		num = tmp;
+	}
+
+	return num;
+}
+
 static struct idlist *add_to_list(struct idlist **root, id_t id, const char *name,
 				  id_t id2, uint16 flags)
 {
@@ -98,7 +116,7 @@ int user_to_uid(const char *name, uid_t *uid_p, BOOL num_ok)
 	if (!name || !*name)
 		return 0;
 	if (num_ok && name[strspn(name, "0123456789")] == '\0') {
-		*uid_p = atol(name);
+		*uid_p = id_parse(name);
 		return 1;
 	}
 	if (!(pass = getpwnam(name)))
@@ -114,7 +132,7 @@ int group_to_gid(const char *name, gid_t *gid_p, BOOL num_ok)
 	if (!name || !*name)
 		return 0;
 	if (num_ok && name[strspn(name, "0123456789")] == '\0') {
-		*gid_p = atol(name);
+		*gid_p = id_parse(name);
 		return 1;
 	}
 	if (!(grp = getgrnam(name)))
@@ -126,12 +144,12 @@ int group_to_gid(const char *name, gid_t *gid_p, BOOL num_ok)
 static int is_in_group(gid_t gid)
 {
 #ifdef HAVE_GETGROUPS
-	static gid_t last_in = GID_NONE, last_out;
-	static int ngroups = -2;
+	static gid_t last_in;
+	static int ngroups = -2, last_out = -1;
 	static GETGROUPS_T *gidset;
 	int n;
 
-	if (gid == last_in)
+	if (gid == last_in && last_out >= 0)
 		return last_out;
 	if (ngroups < -1) {
 		if ((ngroups = getgroups(0, NULL)) < 0)
@@ -230,13 +248,11 @@ static struct idlist *recv_add_id(struct idlist **idlist_ptr, struct idlist *idm
 /* this function is a definate candidate for a faster algorithm */
 uid_t match_uid(uid_t uid)
 {
-	static uid_t last_in = -1, last_out = -1;
+	static struct idlist *last = NULL;
 	struct idlist *list;
 
-	if (uid == last_in)
-		return last_out;
-
-	last_in = uid;
+	if (last && uid == last->id)
+		return last->id2;
 
 	for (list = uidlist; list; list = list->next) {
 		if (list->id == uid)
@@ -245,8 +261,9 @@ uid_t match_uid(uid_t uid)
 
 	if (!list)
 		list = recv_add_id(&uidlist, uidmap, uid, NULL);
+	last = list;
 
-	return last_out = list->id2;
+	return list->id2;
 }
 
 gid_t match_gid(gid_t gid, uint16 *flags_ptr)
@@ -446,11 +463,11 @@ void parse_name_map(char *map, BOOL usernames)
 				exit_cleanup(RERR_SYNTAX);
 			}
 			if (dash)
-				name = (char *)atol(dash+1);
+				name = (char *)id_parse(dash+1);
 			else
 				name = (char *)0;
 			flags = 0;
-			id1 = atol(cp);
+			id1 = id_parse(cp);
 		} else if (strpbrk(cp, "*[?")) {
 			flags = NFLAGS_WILD_NAME_MATCH;
 			name = cp;


-- 
The rsync repository.


More information about the rsync-cvs mailing list