Skip to content

Commit

Permalink
Some uid/gid fixes for (id_t)-1 and other large ID values.
Browse files Browse the repository at this point in the history
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.
  • Loading branch information
Wayne Davison committed Feb 22, 2011
1 parent 7766e67 commit eee2c77
Show file tree
Hide file tree
Showing 2 changed files with 39 additions and 18 deletions.
10 changes: 7 additions & 3 deletions rsync.c
Original file line number Diff line number Diff line change
Expand Up @@ -527,16 +527,20 @@ 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",
change_uid ? "chown" : "chgrp",
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). */
Expand Down
47 changes: 32 additions & 15 deletions uidlist.c
Original file line number Diff line number Diff line change
Expand Up @@ -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)

Expand All @@ -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)
{
Expand Down Expand Up @@ -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)))
Expand All @@ -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)))
Expand All @@ -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)
Expand Down Expand Up @@ -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)
Expand All @@ -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)
Expand Down Expand Up @@ -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;
Expand Down

0 comments on commit eee2c77

Please sign in to comment.