X-Git-Url: https://mattmccutchen.net/rsync/rsync.git/blobdiff_plain/a572e12675882f401e43f231ca4effdedfc70d69..9b25ef35bd8c13480f79753c605f873d9e271936:/flist.c diff --git a/flist.c b/flist.c index c5dc325e..4058aca0 100644 --- a/flist.c +++ b/flist.c @@ -50,8 +50,8 @@ extern int preserve_links; extern int preserve_hard_links; extern int preserve_devices; extern int preserve_specials; -extern int preserve_uid; -extern int preserve_gid; +extern int uid_ndx; +extern int gid_ndx; extern int relative_paths; extern int implied_dirs; extern int file_extra_cnt; @@ -308,6 +308,15 @@ static void flist_expand(struct file_list *flist, int extra) out_of_memory("flist_expand"); } +static void flist_done_allocating(struct file_list *flist) +{ + void *ptr = pool_boundary(flist->file_pool, 8*1024); + if (flist->pool_boundary == ptr) + flist->pool_boundary = NULL; /* list didn't use any pool memory */ + else + flist->pool_boundary = ptr; +} + int push_pathname(const char *dir, int len) { if (dir == pathname) @@ -405,24 +414,24 @@ static void send_file_entry(int f, struct file_struct *file, int ndx) } } else if (protocol_version < 28) rdev = MAKEDEV(0, 0); - if (preserve_uid) { + if (uid_ndx) { if ((uid_t)F_OWNER(file) == uid && *lastname) flags |= XMIT_SAME_UID; else { uid = F_OWNER(file); - if (preserve_uid && !numeric_ids) { + if (uid_ndx && !numeric_ids) { user_name = add_uid(uid); if (inc_recurse && user_name) flags |= XMIT_USER_NAME_FOLLOWS; } } } - if (preserve_gid) { + if (gid_ndx) { if ((gid_t)F_GROUP(file) == gid && *lastname) flags |= XMIT_SAME_GID; else { gid = F_GROUP(file); - if (preserve_gid && !numeric_ids) { + if (gid_ndx && !numeric_ids) { group_name = add_gid(gid); if (inc_recurse && group_name) flags |= XMIT_GROUP_NAME_FOLLOWS; @@ -504,7 +513,7 @@ static void send_file_entry(int f, struct file_struct *file, int ndx) } if (!(flags & XMIT_SAME_MODE)) write_int(f, to_wire_mode(mode)); - if (preserve_uid && !(flags & XMIT_SAME_UID)) { + if (uid_ndx && !(flags & XMIT_SAME_UID)) { if (protocol_version < 30) write_int(f, uid); else { @@ -516,7 +525,7 @@ static void send_file_entry(int f, struct file_struct *file, int ndx) } } } - if (preserve_gid && !(flags & XMIT_SAME_GID)) { + if (gid_ndx && !(flags & XMIT_SAME_GID)) { if (protocol_version < 30) write_int(f, gid); else { @@ -692,9 +701,9 @@ static struct file_struct *recv_file_entry(struct file_list *flist, file_length = F_LENGTH(first); modtime = first->modtime; mode = first->mode; - if (preserve_uid) + if (uid_ndx) uid = F_OWNER(first); - if (preserve_gid) + if (gid_ndx) gid = F_GROUP(first); if ((preserve_devices && IS_DEVICE(mode)) || (preserve_specials && IS_SPECIAL(mode))) { @@ -730,7 +739,7 @@ static struct file_struct *recv_file_entry(struct file_list *flist, if (chmod_modes && !S_ISLNK(mode)) mode = tweak_mode(mode, chmod_modes); - if (preserve_uid && !(xflags & XMIT_SAME_UID)) { + if (uid_ndx && !(xflags & XMIT_SAME_UID)) { if (protocol_version < 30) uid = (uid_t)read_int(f); else { @@ -741,7 +750,7 @@ static struct file_struct *recv_file_entry(struct file_list *flist, uid = match_uid(uid); } } - if (preserve_gid && !(xflags & XMIT_SAME_GID)) { + if (gid_ndx && !(xflags & XMIT_SAME_GID)) { if (protocol_version < 30) gid = (gid_t)read_int(f); else { @@ -848,9 +857,9 @@ static struct file_struct *recv_file_entry(struct file_list *flist, OPT_EXTRA(file, 0)->unum = (uint32)(file_length >> 32); } file->mode = mode; - if (preserve_uid) + if (uid_ndx) F_OWNER(file) = uid; - if (preserve_gid) { + if (gid_ndx) { F_GROUP(file) = gid; file->flags |= gid_flags; } @@ -1169,9 +1178,9 @@ struct file_struct *make_file(const char *fname, struct file_list *flist, OPT_EXTRA(file, 0)->unum = (uint32)(st.st_size >> 32); } file->mode = st.st_mode; - if (preserve_uid) + if (uid_ndx) F_OWNER(file) = st.st_uid; - if (preserve_gid) + if (gid_ndx) F_GROUP(file) = st.st_gid; if (basename != thisname) @@ -1200,9 +1209,9 @@ struct file_struct *make_file(const char *fname, struct file_list *flist, file->modtime = st2.st_mtime; file->len32 = 0; file->mode = st2.st_mode; - if (preserve_uid) + if (uid_ndx) F_OWNER(file) = st2.st_uid; - if (preserve_gid) + if (gid_ndx) F_GROUP(file) = st2.st_gid; } else file->mode = save_mode; @@ -1318,7 +1327,7 @@ static int file_compare(const void *file1, const void *file2) *(struct file_struct **)file2); } -/* The guts of a merge sort algorithm. This was derived from the GNU C +/* The guts of a merge-sort algorithm. This was derived from the glibc * version, but I (Wayne) changed the merge code to do less copying and * to require only half the amount of temporary memory. */ static void fsort_tmp(struct file_struct **fp, size_t num, @@ -1519,55 +1528,34 @@ void send_extra_file_list(int f, int at_least) future_cnt = 0; while (future_cnt < at_least) { struct file_struct *file = dir_flist->sorted[send_dir_ndx]; - int dstart = dir_count; + int dir_ndx, dstart = dir_count; int32 *dp; flist = flist_new(0, "send_extra_file_list"); start_write = stats.total_written; - /* If this is the first of a set of duplicate dirs, we must - * send all the dirs together in a single file-list. We must - * also send the index of the last dir in the header. */ - if (file->flags & FLAG_DUPLICATE) { - int dir_ndx, end_ndx = send_dir_ndx; - struct file_struct *fp = file; - - while (1) { - dp = F_DIRNODE_P(fp); - end_ndx = DIR_NEXT_SIBLING(dp); - fp = dir_flist->sorted[end_ndx]; - if (!(fp->flags & FLAG_DUPLICATE)) - break; - } - #ifdef ICONV_OPTION - if (ic_ndx) - dir_ndx = F_NDX(fp); - else + if (ic_ndx) + dir_ndx = F_NDX(file); + else #endif - dir_ndx = end_ndx; - write_ndx(f, NDX_FLIST_OFFSET - dir_ndx); + dir_ndx = send_dir_ndx; + write_ndx(f, NDX_FLIST_OFFSET - dir_ndx); - while (1) { - send1extra(f, file, flist); - if (send_dir_ndx == end_ndx) - break; - dp = F_DIRNODE_P(file); - send_dir_ndx = DIR_NEXT_SIBLING(dp); - file = dir_flist->sorted[send_dir_ndx]; - } - } else { - int dir_ndx; -#ifdef ICONV_OPTION - if (ic_ndx) - dir_ndx = F_NDX(file); - else -#endif - dir_ndx = send_dir_ndx; - write_ndx(f, NDX_FLIST_OFFSET - dir_ndx); + send1extra(f, file, flist); + dp = F_DIRNODE_P(file); + /* If there are any duplicate directory names that follow, we + * send all the dirs together in one file-list. The dir_flist + * tree links all the child subdirs onto the last dup dir. */ + while ((dir_ndx = DIR_NEXT_SIBLING(dp)) >= 0 + && dir_flist->sorted[dir_ndx]->flags & FLAG_DUPLICATE) { + send_dir_ndx = dir_ndx; + file = dir_flist->sorted[dir_ndx]; send1extra(f, file, flist); + dp = F_DIRNODE_P(file); } + write_byte(f, 0); #ifdef ICONV_OPTION @@ -1583,6 +1571,7 @@ void send_extra_file_list(int f, int at_least) clean_flist(flist, 0); add_dirs_to_tree(send_dir_ndx, flist, dir_count - dstart); + flist_done_allocating(flist); file_total += flist->count; future_cnt += flist->count; @@ -1591,7 +1580,6 @@ void send_extra_file_list(int f, int at_least) if (verbose > 3) output_flist(flist); - dp = F_DIRNODE_P(file); if (DIR_FIRST_CHILD(dp) >= 0) { send_dir_ndx = DIR_FIRST_CHILD(dp); send_dir_depth++; @@ -1917,6 +1905,7 @@ struct file_list *send_file_list(int f, int argc, char *argv[]) if (inc_recurse) { add_dirs_to_tree(-1, flist, dir_count); + flist_done_allocating(flist); if (send_dir_ndx < 0) { write_ndx(f, NDX_FLIST_EOF); flist_eof = 1; @@ -2023,7 +2012,9 @@ struct file_list *recv_file_list(int f) } } - if (!inc_recurse && f >= 0) + if (inc_recurse) + flist_done_allocating(flist); + else if (f >= 0) recv_id_list(f, flist); clean_flist(flist, relative_paths); @@ -2159,34 +2150,42 @@ struct file_list *flist_new(int flags, char *msg) memset(flist, 0, sizeof flist[0]); - if (!(flags & FLIST_TEMP)) { - if (first_flist) { - flist->ndx_start = first_flist->prev->ndx_start - + first_flist->prev->count; - } + if (flags & FLIST_TEMP) { + if (!(flist->file_pool = pool_create(SMALL_EXTENT, 0, + out_of_memory, POOL_INTERN))) + out_of_memory(msg); + } else { /* This is a doubly linked list with prev looping back to * the end of the list, but the last next pointer is NULL. */ - if (!first_flist) + if (!first_flist) { + flist->file_pool = pool_create(NORMAL_EXTENT, 0, + out_of_memory, POOL_INTERN); + if (!flist->file_pool) + out_of_memory(msg); + first_flist = cur_flist = flist->prev = flist; - else { + } else { + flist->file_pool = first_flist->file_pool; + + flist->ndx_start = first_flist->prev->ndx_start + + first_flist->prev->count; + flist->prev = first_flist->prev; flist->prev->next = first_flist->prev = flist; } + flist->pool_boundary = pool_boundary(flist->file_pool, 0); flist_cnt++; } - if (!(flist->file_pool = pool_create(FILE_EXTENT, 0, out_of_memory, POOL_INTERN))) - out_of_memory(msg); - return flist; } /* Free up all elements in a flist. */ void flist_free(struct file_list *flist) { - if (!flist->prev) - ; /* Was FLIST_TEMP dir-list. */ - else if (flist == flist->prev) { + if (!flist->prev) { + /* Was FLIST_TEMP dir-list. */ + } else if (flist == flist->prev) { first_flist = cur_flist = NULL; file_total = 0; flist_cnt = 0; @@ -2205,7 +2204,11 @@ void flist_free(struct file_list *flist) flist_cnt--; } - pool_destroy(flist->file_pool); + if (!flist->prev || !flist_cnt) + pool_destroy(flist->file_pool); + else + pool_free_old(flist->file_pool, flist->pool_boundary); + if (flist->sorted && flist->sorted != flist->files) free(flist->sorted); free(flist->files); @@ -2260,20 +2263,18 @@ static void clean_flist(struct file_list *flist, int strip_root) } else j = -1; if (j >= 0) { - struct file_struct *fp = flist->sorted[j]; int keep, drop; /* If one is a dir and the other is not, we want to * keep the dir because it might have contents in the * list. */ - if (S_ISDIR(file->mode) != S_ISDIR(fp->mode)) { - if (S_ISDIR(file->mode)) + if (S_ISDIR(file->mode)) { + struct file_struct *fp = flist->sorted[j]; + if (!S_ISDIR(fp->mode)) keep = i, drop = j; else keep = j, drop = i; - } else if (protocol_version < 27) + } else keep = j, drop = i; - else - keep = i, drop = j; if (am_sender) flist->sorted[drop]->flags |= FLAG_DUPLICATE; @@ -2399,12 +2400,12 @@ static void output_flist(struct file_list *flist) who, flist->ndx_start, flist->count, flist->low, flist->high); for (i = 0; i < flist->count; i++) { file = flist->sorted[i]; - if ((am_root || am_sender) && preserve_uid) { + if ((am_root || am_sender) && uid_ndx) { snprintf(uidbuf, sizeof uidbuf, " uid=%u", F_OWNER(file)); } else *uidbuf = '\0'; - if (preserve_gid) { + if (gid_ndx) { static char parens[] = "(\0)\0\0\0"; char *pp = parens + (file->flags & FLAG_SKIP_GROUP ? 0 : 3); snprintf(gidbuf, sizeof gidbuf, " gid=%s%u%s",