X-Git-Url: https://mattmccutchen.net/rsync/rsync.git/blobdiff_plain/3ec4dd9781ed23be4a2b939fa5bfded619a1a8fb..3bee67337d0491f55654cf8e926592365e5a8502:/flist.c diff --git a/flist.c b/flist.c index d9f8e440..86646bb5 100644 --- a/flist.c +++ b/flist.c @@ -26,7 +26,7 @@ extern int csum_length; extern int verbose; extern int am_server; extern int always_checksum; -extern off_t total_size; +extern int64 total_size; extern int cvs_exclude; @@ -44,8 +44,9 @@ extern int preserve_times; extern int relative_paths; extern int copy_links; extern int remote_version; +extern int io_error; -static char **local_exclude_list = NULL; +static char **local_exclude_list; int link_stat(const char *Path, struct stat *Buffer) { @@ -68,7 +69,7 @@ static int match_file_name(char *fname,struct stat *st) { if (check_exclude(fname,local_exclude_list)) { if (verbose > 2) - fprintf(FERROR,"excluding file %s\n",fname); + fprintf(FINFO,"excluding file %s\n",fname); return 0; } return 1; @@ -87,7 +88,7 @@ static void set_filesystem(char *fname) static void send_directory(int f,struct file_list *flist,char *dir); -static char *flist_dir = NULL; +static char *flist_dir; static void clean_fname(char *name) { @@ -133,15 +134,15 @@ static void clean_fname(char *name) -void send_file_entry(struct file_struct *file,int f) +void send_file_entry(struct file_struct *file,int f,unsigned base_flags) { unsigned char flags; - static time_t last_time=0; - static mode_t last_mode=0; - static dev_t last_rdev=0; - static uid_t last_uid=0; - static gid_t last_gid=0; - static char lastname[MAXPATHLEN]=""; + static time_t last_time; + static mode_t last_mode; + static dev_t last_rdev; + static uid_t last_uid; + static gid_t last_gid; + static char lastname[MAXPATHLEN]; char *fname; int l1,l2; @@ -154,7 +155,7 @@ void send_file_entry(struct file_struct *file,int f) fname = f_name(file); - flags = FILE_VALID; + flags = base_flags; if (file->mode == last_mode) flags |= SAME_MODE; if (file->rdev == last_rdev) flags |= SAME_RDEV; @@ -177,7 +178,7 @@ void send_file_entry(struct file_struct *file,int f) write_byte(f,l2); write_buf(f,fname+l1,l2); - write_int(f,(int)file->length); + write_longint(f,file->length); if (!(flags & SAME_TIME)) write_int(f,(int)file->modtime); if (!(flags & SAME_MODE)) @@ -202,13 +203,13 @@ void send_file_entry(struct file_struct *file,int f) #if SUPPORT_HARD_LINKS if (preserve_hard_links && S_ISREG(file->mode)) { - write_int(f,file->dev); - write_int(f,file->inode); + write_int(f,(int)file->dev); + write_int(f,(int)file->inode); } #endif if (always_checksum) { - write_buf(f,file->sum,csum_length); + write_buf(f,file->sum,csum_length); } last_mode = file->mode; @@ -223,14 +224,14 @@ void send_file_entry(struct file_struct *file,int f) -void receive_file_entry(struct file_struct **fptr, - unsigned char flags,int f) +static void receive_file_entry(struct file_struct **fptr, + unsigned flags,int f) { - static time_t last_time=0; - static mode_t last_mode=0; - static dev_t last_rdev=0; - static uid_t last_uid=0; - static gid_t last_gid=0; + static time_t last_time; + static mode_t last_mode; + static dev_t last_rdev; + static uid_t last_uid; + static gid_t last_gid; static char lastname[MAXPATHLEN]; char thisname[MAXPATHLEN]; int l1=0,l2=0; @@ -250,8 +251,10 @@ void receive_file_entry(struct file_struct **fptr, bzero((char *)file,sizeof(*file)); (*fptr) = file; + if (l2 >= MAXPATHLEN-l1) overflow("receive_file_entry"); + strncpy(thisname,lastname,l1); - read_buf(f,&thisname[l1],l2); + read_sbuf(f,&thisname[l1],l2); thisname[l1+l2] = 0; strncpy(lastname,thisname,MAXPATHLEN-1); @@ -277,7 +280,8 @@ void receive_file_entry(struct file_struct **fptr, if (!file->basename) out_of_memory("receive_file_entry 1"); - file->length = (off_t)read_int(f); + file->flags = flags; + file->length = read_longint(f); file->modtime = (flags & SAME_TIME) ? last_time : (time_t)read_int(f); file->mode = (flags & SAME_MODE) ? last_mode : (mode_t)read_int(f); if (preserve_uid) @@ -291,8 +295,7 @@ void receive_file_entry(struct file_struct **fptr, int l = read_int(f); file->link = (char *)malloc(l+1); if (!file->link) out_of_memory("receive_file_entry 2"); - read_buf(f,file->link,l); - file->link[l] = 0; + read_sbuf(f,file->link,l); } #if SUPPORT_HARD_LINKS @@ -302,8 +305,11 @@ void receive_file_entry(struct file_struct **fptr, } #endif - if (always_checksum) - read_buf(f,file->sum,csum_length); + if (always_checksum) { + file->sum = (char *)malloc(MD4_SUM_LENGTH); + if (!file->sum) out_of_memory("md4 sum"); + read_buf(f,file->sum,csum_length); + } last_mode = file->mode; last_rdev = file->rdev; @@ -313,6 +319,29 @@ void receive_file_entry(struct file_struct **fptr, } +/* determine if a file in a different filesstem should be skipped + when one_file_system is set. We bascally only want to include + the mount points - but they can be hard to find! */ +static int skip_filesystem(char *fname, struct stat *st) +{ + struct stat st2; + char *p = strrchr(fname, '/'); + + /* skip all but directories */ + if (!S_ISDIR(st->st_mode)) return 1; + + /* if its not a subdirectory then allow */ + if (!p) return 0; + + *p = 0; + if (link_stat(fname, &st2)) { + *p = '/'; + return 0; + } + *p = '/'; + + return (st2.st_dev != filesystem_dev); +} static struct file_struct *make_file(char *fname) { @@ -330,24 +359,27 @@ static struct file_struct *make_file(char *fname) bzero(sum,SUM_LENGTH); if (link_stat(fname,&st) != 0) { + io_error = 1; fprintf(FERROR,"%s: %s\n", fname,strerror(errno)); return NULL; } if (S_ISDIR(st.st_mode) && !recurse) { - fprintf(FERROR,"skipping directory %s\n",fname); + fprintf(FINFO,"skipping directory %s\n",fname); return NULL; } - if (one_file_system && st.st_dev != filesystem_dev) - return NULL; + if (one_file_system && st.st_dev != filesystem_dev) { + if (skip_filesystem(fname, &st)) + return NULL; + } if (!match_file_name(fname,&st)) return NULL; if (verbose > 2) - fprintf(FERROR,"make_file(%s)\n",fname); + fprintf(FINFO,"make_file(%s)\n",fname); file = (struct file_struct *)malloc(sizeof(*file)); if (!file) out_of_memory("make_file"); @@ -385,6 +417,7 @@ static struct file_struct *make_file(char *fname) int l; char lnk[MAXPATHLEN]; if ((l=readlink(fname,lnk,MAXPATHLEN-1)) == -1) { + io_error=1; fprintf(FERROR,"readlink %s : %s\n", fname,strerror(errno)); return NULL; @@ -394,8 +427,17 @@ static struct file_struct *make_file(char *fname) } #endif - if (always_checksum && S_ISREG(st.st_mode)) { - file_checksum(fname,file->sum,st.st_size); + if (always_checksum) { + file->sum = (char *)malloc(MD4_SUM_LENGTH); + if (!file->sum) out_of_memory("md4 sum"); + /* drat. we have to provide a null checksum for non-regular + files in order to be compatible with earlier versions + of rsync */ + if (S_ISREG(st.st_mode)) { + file_checksum(fname,file->sum,st.st_size); + } else { + memset(file->sum, 0, MD4_SUM_LENGTH); + } } if (flist_dir) { @@ -418,7 +460,8 @@ static struct file_struct *make_file(char *fname) -static void send_file_name(int f,struct file_list *flist,char *fname) +static void send_file_name(int f,struct file_list *flist,char *fname, + int recursive, unsigned base_flags) { struct file_struct *file; @@ -440,10 +483,10 @@ static void send_file_name(int f,struct file_list *flist,char *fname) if (strcmp(file->basename,"")) { flist->files[flist->count++] = file; - send_file_entry(file,f); + send_file_entry(file,f,base_flags); } - if (S_ISDIR(file->mode) && recurse) { + if (S_ISDIR(file->mode) && recursive) { char **last_exclude_list = local_exclude_list; send_directory(f,flist,f_name(file)); local_exclude_list = last_exclude_list; @@ -463,6 +506,7 @@ static void send_directory(int f,struct file_list *flist,char *dir) d = opendir(dir); if (!d) { + io_error = 1; fprintf(FERROR,"%s: %s\n", dir,strerror(errno)); return; @@ -473,6 +517,7 @@ static void send_directory(int f,struct file_list *flist,char *dir) l = strlen(fname); if (fname[l-1] != '/') { if (l == MAXPATHLEN-1) { + io_error = 1; fprintf(FERROR,"skipping long-named directory %s\n",fname); closedir(d); return; @@ -487,7 +532,8 @@ static void send_directory(int f,struct file_list *flist,char *dir) strcpy(p,".cvsignore"); local_exclude_list = make_exclude_list(fname,NULL,0); } else { - fprintf(FERROR,"cannot cvs-exclude in long-named directory %s\n",fname); + io_error = 1; + fprintf(FINFO,"cannot cvs-exclude in long-named directory %s\n",fname); } } @@ -496,7 +542,7 @@ static void send_directory(int f,struct file_list *flist,char *dir) strcmp(di->d_name,"..")==0) continue; strncpy(p,di->d_name,MAXPATHLEN-(l+1)); - send_file_name(f,flist,fname); + send_file_name(f,flist,fname,recurse,FLAG_DELETE); } closedir(d); @@ -506,107 +552,136 @@ static void send_directory(int f,struct file_list *flist,char *dir) struct file_list *send_file_list(int f,int argc,char *argv[]) { - int i,l; - struct stat st; - char *p,*dir; - char dbuf[MAXPATHLEN]; - struct file_list *flist; - - if (verbose && recurse && !am_server && f != -1) { - fprintf(FINFO,"building file list ... "); - fflush(FINFO); - } - - flist = (struct file_list *)malloc(sizeof(flist[0])); - if (!flist) out_of_memory("send_file_list"); - - flist->count=0; - flist->malloced = 1000; - flist->files = (struct file_struct **)malloc(sizeof(flist->files[0])* - flist->malloced); - if (!flist->files) out_of_memory("send_file_list"); + int i,l; + struct stat st; + char *p,*dir; + char dbuf[MAXPATHLEN]; + char lastpath[MAXPATHLEN]=""; + struct file_list *flist; + + if (verbose && recurse && !am_server && f != -1) { + fprintf(FINFO,"building file list ... "); + fflush(FINFO); + } - for (i=0;icount=0; + flist->malloced = 1000; + flist->files = (struct file_struct **)malloc(sizeof(flist->files[0])* + flist->malloced); + if (!flist->files) out_of_memory("send_file_list"); - l = strlen(fname); - if (l != 1 && fname[l-1] == '/') { - strcat(fname,"."); - } + for (i=0;i= 15) { + send_uid_list(f); + } - clean_flist(flist); + /* if protocol version is >= 17 then send the io_error flag */ + if (f != -1 && remote_version >= 17) { + write_int(f, io_error); + } - /* now send the uid/gid list. This was introduced in protocol version 15 */ - if (f != -1 && remote_version >= 15) { - send_uid_list(f); - } + if (verbose > 2) + fprintf(FINFO,"send_file_list done\n"); - return flist; + return flist; } @@ -655,12 +730,12 @@ struct file_list *recv_file_list(int f) flist->count++; if (verbose > 2) - fprintf(FERROR,"recv_file_name(%s)\n",f_name(flist->files[i])); + fprintf(FINFO,"recv_file_name(%s)\n",f_name(flist->files[i])); } if (verbose > 2) - fprintf(FERROR,"received %d names\n",flist->count); + fprintf(FINFO,"received %d names\n",flist->count); clean_flist(flist); @@ -673,6 +748,14 @@ struct file_list *recv_file_list(int f) recv_uid_list(f, flist); } + /* if protocol version is >= 17 then recv the io_error flag */ + if (f != -1 && remote_version >= 17) { + io_error |= read_int(f); + } + + if (verbose > 2) + fprintf(FINFO,"recv_file_list done\n"); + return flist; oom: @@ -723,8 +806,8 @@ static void free_file(struct file_struct *file) if (!file) return; if (file->basename) free(file->basename); if (file->link) free(file->link); + if (file->sum) free(file->sum); bzero((char *)file, sizeof(*file)); - free(file); } @@ -736,6 +819,7 @@ void flist_free(struct file_list *flist) int i; for (i=1;icount;i++) { free_file(flist->files[i]); + free(flist->files[i]); } bzero((char *)flist->files, sizeof(flist->files[0])*flist->count); free(flist->files); @@ -761,13 +845,13 @@ void clean_flist(struct file_list *flist) for (i=1;icount;i++) { if (flist->files[i]->basename && + flist->files[i-1]->basename && strcmp(f_name(flist->files[i]), f_name(flist->files[i-1])) == 0) { if (verbose > 1 && !am_server) - fprintf(FERROR,"removing duplicate name %s from file list %d\n", + fprintf(FINFO,"removing duplicate name %s from file list %d\n", f_name(flist->files[i-1]),i-1); free_file(flist->files[i]); - flist->files[i] = NULL; } } } @@ -782,7 +866,7 @@ char *f_name(struct file_struct *f) static int n; char *p = names[n]; - if (!f) return NULL; + if (!f || !f->basename) return NULL; n = (n+1)%10;