- Got rid of a bunch of XFLG_* options.
[rsync/rsync.git] / options.c
CommitLineData
7a24c346 1/* -*- c-file-style: "linux" -*-
dfa32483 2 *
dafe63ca
MP
3 * Copyright (C) 1998-2001 by Andrew Tridgell <tridge@samba.org>
4 * Copyright (C) 2000, 2001, 2002 by Martin Pool <mbp@samba.org>
dfa32483 5 *
dafe63ca
MP
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation; either version 2 of the License, or
9 * (at your option) any later version.
dfa32483 10 *
dafe63ca
MP
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
dfa32483 15 *
dafe63ca
MP
16 * You should have received a copy of the GNU General Public License
17 * along with this program; if not, write to the Free Software
18 * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
19 */
7a6421fa 20
7a6421fa 21#include "rsync.h"
2855f61f 22#include "popt.h"
7a6421fa 23
7bc90b30 24extern int module_id;
7be73df4 25extern int sanitize_paths;
99218d82 26extern int select_timeout;
7842418b
WD
27extern struct filter_list_struct filter_list;
28extern struct filter_list_struct server_filter_list;
8645af1d 29
7a6421fa 30int make_backups = 0;
1bfbf40b
MP
31
32/**
dfa32483 33 * If 1, send the whole file as literal data rather than trying to
dafe63ca 34 * create an incremental diff.
1bfbf40b 35 *
dfa32483 36 * If -1, then look at whether we're local or remote and go by that.
dafe63ca
MP
37 *
38 * @sa disable_deltas_p()
1bfbf40b 39 **/
dfa32483 40int whole_file = -1;
1bfbf40b 41
dfa32483 42int archive_mode = 0;
716e73d4 43int keep_dirlinks = 0;
7a6421fa
AT
44int copy_links = 0;
45int preserve_links = 0;
46int preserve_hard_links = 0;
47int preserve_perms = 0;
48int preserve_devices = 0;
49int preserve_uid = 0;
50int preserve_gid = 0;
51int preserve_times = 0;
20fb7b91 52int omit_dir_times = 0;
7a6421fa
AT
53int update_only = 0;
54int cvs_exclude = 0;
a5c11139
WD
55int dry_run = 0;
56int local_server = 0;
57int ignore_times = 0;
58int delete_mode = 0;
a51b3168 59int delete_during = 0;
51d48398
WD
60int delete_before = 0;
61int delete_after = 0;
a5c11139
WD
62int delete_excluded = 0;
63int one_file_system = 0;
4f3e9a0f 64int protocol_version = PROTOCOL_VERSION;
a5c11139
WD
65int sparse_files = 0;
66int do_compression = 0;
67int am_root = 0;
68int orig_umask = 0;
ea5164d1
WD
69int relative_paths = -1;
70int implied_dirs = 1;
7a6421fa
AT
71int numeric_ids = 0;
72int force_delete = 0;
73int io_timeout = 0;
7a6421fa 74int am_server = 0;
f1d5ba40 75int am_sender = -1;
4337c8f8 76int am_generator = 0;
ea5164d1
WD
77char *files_from = NULL;
78int filesfrom_fd = -1;
79char *remote_filesfrom_file = NULL;
80int eol_nulls = 0;
7a6421fa 81int recurse = 0;
65e4cda0 82int xfer_dirs = 0;
1312d9fc
WD
83int am_daemon = 0;
84int daemon_over_rsh = 0;
a5c11139
WD
85int do_stats = 0;
86int do_progress = 0;
87int keep_partial = 0;
88int safe_symlinks = 0;
89int copy_unsafe_links = 0;
90int size_only = 0;
9fb08441 91int daemon_bwlimit = 0;
a5c11139 92int bwlimit = 0;
3c74c3a3 93size_t bwlimit_writemax = 0;
a5c11139
WD
94int only_existing = 0;
95int opt_ignore_existing = 0;
96int max_delete = 0;
7d5acf1d 97OFF_T max_size = 0;
a5c11139
WD
98int ignore_errors = 0;
99int modify_window = 0;
100int blocking_io = -1;
2289bf64 101int checksum_seed = 0;
a3221d2a 102int inplace = 0;
f06e7082 103int delay_updates = 0;
a06b419d 104long block_size = 0; /* "long" because popt can't set an int32. */
7a6421fa 105
b35d0d8e 106
13e29995 107/** Network address family. **/
25ff0441 108#if INET6
13e29995 109int default_af_hint = 0; /* Any protocol */
6ab6d4bf 110#else
13e29995 111int default_af_hint = AF_INET; /* Must use IPv4 */
6ab6d4bf 112#endif
06963d0f 113
13e29995
MP
114/** Do not go into the background when run as --daemon. Good
115 * for debugging and required for running as a service on W32,
116 * or under Unix process-monitors. **/
117int no_detach = 0;
118
088aac85
DD
119int write_batch = 0;
120int read_batch = 0;
d175d7e1
WD
121int backup_dir_len = 0;
122int backup_suffix_len;
e0391f81 123unsigned int backup_dir_remainder;
6902ed17 124
d175d7e1 125char *backup_suffix = NULL;
7a6421fa 126char *tmpdir = NULL;
a7260c40 127char *partial_dir = NULL;
e012f858 128char *basis_dir[MAX_BASIS_DIRS+1];
30e8c8e1 129char *config_file = NULL;
7a6421fa 130char *shell_cmd = NULL;
b6062654 131char *log_format = NULL;
65575e96 132char *password_file = NULL;
41bd28fe 133char *rsync_path = RSYNC_PATH;
66203a98 134char *backup_dir = NULL;
e0391f81 135char backup_dir_buf[MAXPATHLEN];
0c56b1ad 136int rsync_port = 0;
dfd7d541 137int compare_dest = 0;
e012f858 138int copy_dest = 0;
59c95e42 139int link_dest = 0;
ce0b384f 140int basis_dir_cnt = 0;
7a6421fa
AT
141
142int verbose = 0;
b86f0cef 143int quiet = 0;
7a6421fa 144int always_checksum = 0;
f7632fc6 145int list_only = 0;
7a6421fa 146
9b3318b0
WD
147#define MAX_BATCH_NAME_LEN 256 /* Must be less than MAXPATHLEN-13 */
148char *batch_name = NULL;
6902ed17 149
e1add893 150static int daemon_opt; /* sets am_daemon after option error-reporting */
aa4d3b4c 151static int F_option_cnt = 0;
5b56cc19 152static int modify_window_set;
12a79db2 153static char *dest_option = NULL;
7d5acf1d 154static char *max_size_arg;
5b56cc19 155
06963d0f 156/** Local address to bind. As a character string because it's
fdf57ede 157 * interpreted by the IPv6 layer: should be a numeric IP4 or IP6
06963d0f
MP
158 * address, or a hostname. **/
159char *bind_address;
5c9730a4 160
7a24c346 161
27a12348 162static void print_rsync_version(enum logcode f)
7a24c346 163{
dfa32483 164 char const *got_socketpair = "no ";
a3221d2a 165 char const *have_inplace = "no ";
dfa32483
WD
166 char const *hardlinks = "no ";
167 char const *links = "no ";
a358449a 168 char const *ipv6 = "no ";
736a6a29 169 STRUCT_STAT *dumstat;
0c80cd8e 170
25ff0441 171#if HAVE_SOCKETPAIR
dfa32483 172 got_socketpair = "";
0c80cd8e 173#endif
2855f61f 174
a3221d2a
WD
175#if HAVE_FTRUNCATE
176 have_inplace = "";
177#endif
178
2855f61f 179#if SUPPORT_HARD_LINKS
dfa32483 180 hardlinks = "";
2855f61f
MP
181#endif
182
183#if SUPPORT_LINKS
dfa32483 184 links = "";
2855f61f
MP
185#endif
186
a358449a
MP
187#if INET6
188 ipv6 = "";
dfa32483 189#endif
a358449a 190
dfa32483
WD
191 rprintf(f, "%s version %s protocol version %d\n",
192 RSYNC_NAME, RSYNC_VERSION, PROTOCOL_VERSION);
193 rprintf(f,
72a90c75 194 "Copyright (C) 1996-2005 by Andrew Tridgell and others\n");
3b4b1984 195 rprintf(f, "<http://rsync.samba.org/>\n");
dfa32483
WD
196 rprintf(f, "Capabilities: %d-bit files, %ssocketpairs, "
197 "%shard links, %ssymlinks, batchfiles, \n",
a5c11139 198 (int) (sizeof (OFF_T) * 8),
dfa32483 199 got_socketpair, hardlinks, links);
2855f61f 200
736a6a29
MP
201 /* Note that this field may not have type ino_t. It depends
202 * on the complicated interaction between largefile feature
203 * macros. */
a3221d2a
WD
204 rprintf(f, " %sinplace, %sIPv6, %d-bit system inums, %d-bit internal inums\n",
205 have_inplace, ipv6,
a5c11139 206 (int) (sizeof dumstat->st_ino * 8),
1490812a 207 (int) (sizeof (int64) * 8));
fc0302cf
MP
208#ifdef MAINTAINER_MODE
209 rprintf(f, " panic action: \"%s\"\n",
210 get_panic_action());
211#endif
736a6a29 212
031fa9ad
WD
213#if SIZEOF_INT64 < 8
214 rprintf(f, "WARNING: no 64-bit integers on this platform!\n");
7a24c346 215#endif
c561edaa
WD
216 if (sizeof (int64) != SIZEOF_INT64) {
217 rprintf(f,
218 "WARNING: size mismatch in SIZEOF_INT64 define (%d != %d)\n",
219 (int) SIZEOF_INT64, (int) sizeof (int64));
220 }
7b329a2d
MP
221
222 rprintf(f,
223"\n"
224"rsync comes with ABSOLUTELY NO WARRANTY. This is free software, and you\n"
225"are welcome to redistribute it under certain conditions. See the GNU\n"
226"General Public Licence for details.\n"
227 );
7a24c346
MP
228}
229
230
0f3203c3 231void usage(enum logcode F)
7a6421fa 232{
2855f61f 233 print_rsync_version(F);
704f908e 234
3ff984d7 235 rprintf(F,"\nrsync is a file transfer program capable of efficient remote update\nvia a fast differencing algorithm.\n\n");
704f908e 236
9ef53907 237 rprintf(F,"Usage: rsync [OPTION]... SRC [SRC]... [USER@]HOST:DEST\n");
704f908e 238 rprintf(F," or rsync [OPTION]... [USER@]HOST:SRC DEST\n");
9ef53907 239 rprintf(F," or rsync [OPTION]... SRC [SRC]... DEST\n");
14d43f1f 240 rprintf(F," or rsync [OPTION]... [USER@]HOST::SRC [DEST]\n");
9ef53907 241 rprintf(F," or rsync [OPTION]... SRC [SRC]... [USER@]HOST::DEST\n");
14d43f1f 242 rprintf(F," or rsync [OPTION]... rsync://[USER@]HOST[:PORT]/SRC [DEST]\n");
eaa4c150 243 rprintf(F," or rsync [OPTION]... SRC [SRC]... rsync://[USER@]HOST[:PORT]/DEST\n");
9ef53907
DD
244 rprintf(F,"SRC on single-colon remote HOST will be expanded by remote shell\n");
245 rprintf(F,"SRC on server remote HOST may contain shell wildcards or multiple\n");
246 rprintf(F," sources separated by space as long as they have same top-level\n");
704f908e
AT
247 rprintf(F,"\nOptions\n");
248 rprintf(F," -v, --verbose increase verbosity\n");
b3708acf
WD
249 rprintf(F," -q, --quiet suppress non-error messages\n");
250 rprintf(F," -c, --checksum skip based on checksum, not mod-time & size\n");
251 rprintf(F," -a, --archive archive mode; same as -rlptgoD (no -H)\n");
704f908e
AT
252 rprintf(F," -r, --recursive recurse into directories\n");
253 rprintf(F," -R, --relative use relative path names\n");
ea5164d1
WD
254 rprintf(F," --no-relative turn off --relative\n");
255 rprintf(F," --no-implied-dirs don't send implied dirs with -R\n");
6839140e 256 rprintf(F," -b, --backup make backups (see --suffix & --backup-dir)\n");
b3708acf
WD
257 rprintf(F," --backup-dir=DIR make backups into hierarchy based in DIR\n");
258 rprintf(F," --suffix=SUFFIX set backup suffix (default %s w/o --backup-dir)\n",BACKUP_SUFFIX);
259 rprintf(F," -u, --update skip files that are newer on the receiver\n");
4ce838e1 260 rprintf(F," --inplace update destination files in-place (SEE MAN PAGE)\n");
65e4cda0 261 rprintf(F," -d, --dirs transfer directories without recursing\n");
13e29995 262 rprintf(F," -l, --links copy symlinks as symlinks\n");
b3708acf
WD
263 rprintf(F," -L, --copy-links transform symlink into referent file/dir\n");
264 rprintf(F," --copy-unsafe-links only \"unsafe\" symlinks are transformed\n");
265 rprintf(F," --safe-links ignore symlinks that point outside the source tree\n");
704f908e 266 rprintf(F," -H, --hard-links preserve hard links\n");
65e4cda0 267 rprintf(F," -K, --keep-dirlinks treat symlinked dir on receiver as dir\n");
704f908e
AT
268 rprintf(F," -p, --perms preserve permissions\n");
269 rprintf(F," -o, --owner preserve owner (root only)\n");
270 rprintf(F," -g, --group preserve group\n");
271 rprintf(F," -D, --devices preserve devices (root only)\n");
dfa32483 272 rprintf(F," -t, --times preserve times\n");
20fb7b91 273 rprintf(F," -O, --omit-dir-times omit directories when preserving times\n");
704f908e
AT
274 rprintf(F," -S, --sparse handle sparse files efficiently\n");
275 rprintf(F," -n, --dry-run show what would have been transferred\n");
b3708acf
WD
276 rprintf(F," -W, --whole-file copy files whole\n");
277 rprintf(F," --no-whole-file always use incremental rsync algorithm\n");
704f908e 278 rprintf(F," -x, --one-file-system don't cross filesystem boundaries\n");
9cd339eb 279 rprintf(F," -B, --block-size=SIZE force a fixed checksum block-size\n");
b3708acf 280 rprintf(F," -e, --rsh=COMMAND specify the remote shell to use\n");
704f908e 281 rprintf(F," --rsync-path=PATH specify path to rsync on the remote machine\n");
b3708acf 282 rprintf(F," --existing only update files that already exist on receiver\n");
6839140e 283 rprintf(F," --ignore-existing ignore files that already exist on receiving side\n");
3359acb8 284 rprintf(F," --del an alias for --delete-during\n");
704f908e 285 rprintf(F," --delete delete files that don't exist on the sending side\n");
c6eb7fad 286 rprintf(F," --delete-before receiver deletes before transfer (default)\n");
3359acb8
WD
287 rprintf(F," --delete-during receiver deletes during transfer, not before\n");
288 rprintf(F," --delete-after receiver deletes after transfer, not before\n");
51d48398 289 rprintf(F," --delete-excluded also delete excluded files on the receiving side\n");
db2b5cb7 290 rprintf(F," --ignore-errors delete even if there are I/O errors\n");
51d48398 291 rprintf(F," --force force deletion of directories even if not empty\n");
0b73ca12 292 rprintf(F," --max-delete=NUM don't delete more than NUM files\n");
7d5acf1d 293 rprintf(F," --max-size=SIZE don't transfer any file larger than SIZE\n");
c95da96a 294 rprintf(F," --partial keep partially transferred files\n");
a7260c40 295 rprintf(F," --partial-dir=DIR put a partially transferred file into DIR\n");
b3708acf 296 rprintf(F," --delay-updates put all updated files into place at transfer's end\n");
704f908e 297 rprintf(F," --numeric-ids don't map uid/gid values by user/group name\n");
db2b5cb7 298 rprintf(F," --timeout=TIME set I/O timeout in seconds\n");
b3708acf
WD
299 rprintf(F," -I, --ignore-times don't skip files that match in size and mod-time\n");
300 rprintf(F," --size-only skip files that match in size\n");
301 rprintf(F," --modify-window=NUM compare mod-times with reduced accuracy\n");
4db88e5b 302 rprintf(F," -T, --temp-dir=DIR create temporary files in directory DIR\n");
375a4556 303 rprintf(F," --compare-dest=DIR also compare destination files relative to DIR\n");
e012f858
WD
304 rprintf(F," --copy-dest=DIR ... and include copies of unchanged files\n");
305 rprintf(F," --link-dest=DIR hardlink to files in DIR when unchanged\n");
704f908e 306 rprintf(F," -z, --compress compress file data\n");
b3708acf 307 rprintf(F," -C, --cvs-exclude auto-ignore files the same way CVS does\n");
aa4d3b4c
WD
308 rprintf(F," -f, --filter=RULE add a file-filtering RULE\n");
309 rprintf(F," -F same as --filter=': /.rsync-filter'\n");
310 rprintf(F," repeated: --filter='- .rsync-filter'\n");
2acf81eb 311 rprintf(F," --exclude=PATTERN exclude files matching PATTERN\n");
b3708acf 312 rprintf(F," --exclude-from=FILE read exclude patterns from FILE\n");
2acf81eb 313 rprintf(F," --include=PATTERN don't exclude files matching PATTERN\n");
b3708acf
WD
314 rprintf(F," --include-from=FILE read include patterns from FILE\n");
315 rprintf(F," --files-from=FILE read list of source-file names from FILE\n");
4db88e5b 316 rprintf(F," -0, --from0 all *-from file lists are delimited by nulls\n");
dfa32483 317 rprintf(F," --version print version number\n");
b4713295 318 rprintf(F," --port=PORT specify double-colon alternate port number\n");
db2b5cb7 319 rprintf(F," --blocking-io use blocking I/O for the remote shell\n");
b3708acf
WD
320 rprintf(F," --no-blocking-io turn off blocking I/O when it is the default\n");
321 rprintf(F," --stats give some file-transfer stats\n");
dfa32483 322 rprintf(F," --progress show progress during transfer\n");
b3708acf
WD
323 rprintf(F," -P same as --partial --progress\n");
324 rprintf(F," --log-format=FORMAT log file-transfers using specified format\n");
325 rprintf(F," --password-file=FILE read password from FILE\n");
65e4cda0 326 rprintf(F," --list-only list the files instead of copying them\n");
b3708acf
WD
327 rprintf(F," --bwlimit=KBPS limit I/O bandwidth; KBytes per second\n");
328 rprintf(F," --write-batch=FILE write a batched update to FILE\n");
329 rprintf(F," --read-batch=FILE read a batched update from FILE\n");
25ff0441 330#if INET6
4db88e5b
WD
331 rprintf(F," -4, --ipv4 prefer IPv4\n");
332 rprintf(F," -6, --ipv6 prefer IPv6\n");
06963d0f 333#endif
3dd22903 334 rprintf(F," -h, --help show this help screen\n");
7a6421fa 335
3ac7f5d4
WD
336 rprintf(F,"\nUse \"rsync --daemon --help\" to see the daemon-mode command-line options.\n");
337 rprintf(F,"Please see the rsync(1) and rsyncd.conf(5) man pages for full documentation.\n");
2855f61f 338 rprintf(F,"See http://rsync.samba.org/ for updates, bug reports, and answers\n");
7a6421fa
AT
339}
340
3ac7f5d4 341enum {OPT_VERSION = 1000, OPT_DAEMON, OPT_SENDER, OPT_EXCLUDE, OPT_EXCLUDE_FROM,
aa4d3b4c 342 OPT_FILTER, OPT_COMPARE_DEST, OPT_COPY_DEST, OPT_LINK_DEST,
1f3d6cdd 343 OPT_INCLUDE, OPT_INCLUDE_FROM, OPT_MODIFY_WINDOW,
7d5acf1d 344 OPT_READ_BATCH, OPT_WRITE_BATCH, OPT_TIMEOUT, OPT_MAX_SIZE,
c67d1386 345 OPT_REFUSED_BASE = 9000};
7a6421fa 346
2855f61f
MP
347static struct poptOption long_options[] = {
348 /* longName, shortName, argInfo, argPtr, value, descrip, argDesc */
8db7cc2c 349 {"version", 0, POPT_ARG_NONE, 0, OPT_VERSION, 0, 0},
d175d7e1 350 {"suffix", 0, POPT_ARG_STRING, &backup_suffix, 0, 0, 0 },
3dd22903
WD
351 {"rsync-path", 0, POPT_ARG_STRING, &rsync_path, 0, 0, 0 },
352 {"password-file", 0, POPT_ARG_STRING, &password_file, 0, 0, 0 },
afb6e945
WD
353 {"ignore-times", 'I', POPT_ARG_NONE, &ignore_times, 0, 0, 0 },
354 {"size-only", 0, POPT_ARG_NONE, &size_only, 0, 0, 0 },
8b54f004 355 {"modify-window", 0, POPT_ARG_INT, &modify_window, OPT_MODIFY_WINDOW, 0, 0 },
afb6e945 356 {"one-file-system", 'x', POPT_ARG_NONE, &one_file_system, 0, 0, 0 },
afb6e945
WD
357 {"existing", 0, POPT_ARG_NONE, &only_existing, 0, 0, 0 },
358 {"ignore-existing", 0, POPT_ARG_NONE, &opt_ignore_existing, 0, 0, 0 },
3359acb8 359 {"delete", 0, POPT_ARG_NONE, &delete_mode, 0, 0, 0 },
c6eb7fad 360 {"delete-before", 0, POPT_ARG_VAL, &delete_before, 2, 0, 0 },
3359acb8 361 {"delete-during", 0, POPT_ARG_NONE, &delete_during, 0, 0, 0 },
51d48398
WD
362 {"delete-after", 0, POPT_ARG_NONE, &delete_after, 0, 0, 0 },
363 {"delete-excluded", 0, POPT_ARG_NONE, &delete_excluded, 0, 0, 0 },
afb6e945
WD
364 {"force", 0, POPT_ARG_NONE, &force_delete, 0, 0, 0 },
365 {"numeric-ids", 0, POPT_ARG_NONE, &numeric_ids, 0, 0, 0 },
aa4d3b4c 366 {"filter", 'f', POPT_ARG_STRING, 0, OPT_FILTER, 0, 0 },
8db7cc2c
WD
367 {"exclude", 0, POPT_ARG_STRING, 0, OPT_EXCLUDE, 0, 0 },
368 {"include", 0, POPT_ARG_STRING, 0, OPT_INCLUDE, 0, 0 },
369 {"exclude-from", 0, POPT_ARG_STRING, 0, OPT_EXCLUDE_FROM, 0, 0 },
370 {"include-from", 0, POPT_ARG_STRING, 0, OPT_INCLUDE_FROM, 0, 0 },
afb6e945 371 {"safe-links", 0, POPT_ARG_NONE, &safe_symlinks, 0, 0, 0 },
8db7cc2c 372 {"help", 'h', POPT_ARG_NONE, 0, 'h', 0, 0 },
afb6e945
WD
373 {"backup", 'b', POPT_ARG_NONE, &make_backups, 0, 0, 0 },
374 {"dry-run", 'n', POPT_ARG_NONE, &dry_run, 0, 0, 0 },
375 {"sparse", 'S', POPT_ARG_NONE, &sparse_files, 0, 0, 0 },
376 {"cvs-exclude", 'C', POPT_ARG_NONE, &cvs_exclude, 0, 0, 0 },
377 {"update", 'u', POPT_ARG_NONE, &update_only, 0, 0, 0 },
a3221d2a 378 {"inplace", 0, POPT_ARG_NONE, &inplace, 0, 0, 0 },
65e4cda0 379 {"dirs", 'd', POPT_ARG_VAL, &xfer_dirs, 2, 0, 0 },
afb6e945
WD
380 {"links", 'l', POPT_ARG_NONE, &preserve_links, 0, 0, 0 },
381 {"copy-links", 'L', POPT_ARG_NONE, &copy_links, 0, 0, 0 },
65e4cda0 382 {"keep-dirlinks", 'K', POPT_ARG_NONE, &keep_dirlinks, 0, 0, 0 },
afb6e945
WD
383 {"whole-file", 'W', POPT_ARG_VAL, &whole_file, 1, 0, 0 },
384 {"no-whole-file", 0, POPT_ARG_VAL, &whole_file, 0, 0, 0 },
385 {"copy-unsafe-links", 0, POPT_ARG_NONE, &copy_unsafe_links, 0, 0, 0 },
386 {"perms", 'p', POPT_ARG_NONE, &preserve_perms, 0, 0, 0 },
387 {"owner", 'o', POPT_ARG_NONE, &preserve_uid, 0, 0, 0 },
388 {"group", 'g', POPT_ARG_NONE, &preserve_gid, 0, 0, 0 },
389 {"devices", 'D', POPT_ARG_NONE, &preserve_devices, 0, 0, 0 },
390 {"times", 't', POPT_ARG_NONE, &preserve_times, 0, 0, 0 },
20fb7b91 391 {"omit-dir-times", 'O', POPT_ARG_NONE, &omit_dir_times, 0, 0, 0 },
afb6e945 392 {"checksum", 'c', POPT_ARG_NONE, &always_checksum, 0, 0, 0 },
8db7cc2c
WD
393 {"verbose", 'v', POPT_ARG_NONE, 0, 'v', 0, 0 },
394 {"quiet", 'q', POPT_ARG_NONE, 0, 'q', 0, 0 },
dfa32483 395 {"archive", 'a', POPT_ARG_NONE, &archive_mode, 0, 0, 0 },
afb6e945 396 {"server", 0, POPT_ARG_NONE, &am_server, 0, 0, 0 },
8db7cc2c 397 {"sender", 0, POPT_ARG_NONE, 0, OPT_SENDER, 0, 0 },
51d48398
WD
398 {"recursive", 'r', POPT_ARG_VAL, &recurse, -1, 0, 0 },
399 {"list-only", 0, POPT_ARG_VAL, &list_only, 2, 0, 0 },
ea5164d1
WD
400 {"relative", 'R', POPT_ARG_VAL, &relative_paths, 1, 0, 0 },
401 {"no-relative", 0, POPT_ARG_VAL, &relative_paths, 0, 0, 0 },
afb6e945 402 {"rsh", 'e', POPT_ARG_STRING, &shell_cmd, 0, 0, 0 },
a06b419d 403 {"block-size", 'B', POPT_ARG_LONG, &block_size, 0, 0, 0 },
afb6e945 404 {"max-delete", 0, POPT_ARG_INT, &max_delete, 0, 0, 0 },
7d5acf1d 405 {"max-size", 0, POPT_ARG_STRING, &max_size_arg, OPT_MAX_SIZE, 0, 0 },
99218d82 406 {"timeout", 0, POPT_ARG_INT, &io_timeout, OPT_TIMEOUT, 0, 0 },
afb6e945 407 {"temp-dir", 'T', POPT_ARG_STRING, &tmpdir, 0, 0, 0 },
e012f858
WD
408 {"compare-dest", 0, POPT_ARG_STRING, 0, OPT_COMPARE_DEST, 0, 0 },
409 {"copy-dest", 0, POPT_ARG_STRING, 0, OPT_COPY_DEST, 0, 0 },
410 {"link-dest", 0, POPT_ARG_STRING, 0, OPT_LINK_DEST, 0, 0 },
2855f61f 411 /* TODO: Should this take an optional int giving the compression level? */
afb6e945 412 {"compress", 'z', POPT_ARG_NONE, &do_compression, 0, 0, 0 },
afb6e945
WD
413 {"stats", 0, POPT_ARG_NONE, &do_stats, 0, 0, 0 },
414 {"progress", 0, POPT_ARG_NONE, &do_progress, 0, 0, 0 },
415 {"partial", 0, POPT_ARG_NONE, &keep_partial, 0, 0, 0 },
a7260c40 416 {"partial-dir", 0, POPT_ARG_STRING, &partial_dir, 0, 0, 0 },
f06e7082 417 {"delay-updates", 0, POPT_ARG_NONE, &delay_updates, 0, 0, 0 },
afb6e945
WD
418 {"ignore-errors", 0, POPT_ARG_NONE, &ignore_errors, 0, 0, 0 },
419 {"blocking-io", 0, POPT_ARG_VAL, &blocking_io, 1, 0, 0 },
420 {"no-blocking-io", 0, POPT_ARG_VAL, &blocking_io, 0, 0, 0 },
aa4d3b4c 421 {0, 'F', POPT_ARG_NONE, 0, 'F', 0, 0 },
8db7cc2c 422 {0, 'P', POPT_ARG_NONE, 0, 'P', 0, 0 },
b4713295 423 {"port", 0, POPT_ARG_INT, &rsync_port, 0, 0, 0 },
afb6e945
WD
424 {"log-format", 0, POPT_ARG_STRING, &log_format, 0, 0, 0 },
425 {"bwlimit", 0, POPT_ARG_INT, &bwlimit, 0, 0, 0 },
afb6e945
WD
426 {"backup-dir", 0, POPT_ARG_STRING, &backup_dir, 0, 0, 0 },
427 {"hard-links", 'H', POPT_ARG_NONE, &preserve_hard_links, 0, 0, 0 },
8db7cc2c
WD
428 {"read-batch", 0, POPT_ARG_STRING, &batch_name, OPT_READ_BATCH, 0, 0 },
429 {"write-batch", 0, POPT_ARG_STRING, &batch_name, OPT_WRITE_BATCH, 0, 0 },
ea5164d1
WD
430 {"files-from", 0, POPT_ARG_STRING, &files_from, 0, 0, 0 },
431 {"from0", '0', POPT_ARG_NONE, &eol_nulls, 0, 0, 0},
432 {"no-implied-dirs", 0, POPT_ARG_VAL, &implied_dirs, 0, 0, 0 },
4f3e9a0f 433 {"protocol", 0, POPT_ARG_INT, &protocol_version, 0, 0, 0 },
c8d895de 434 {"checksum-seed", 0, POPT_ARG_INT, &checksum_seed, 0, 0, 0 },
25ff0441 435#if INET6
3dd22903
WD
436 {"ipv4", '4', POPT_ARG_VAL, &default_af_hint, AF_INET, 0, 0 },
437 {"ipv6", '6', POPT_ARG_VAL, &default_af_hint, AF_INET6, 0, 0 },
06963d0f 438#endif
3ac7f5d4
WD
439 /* All these options switch us into daemon-mode option-parsing. */
440 {"address", 0, POPT_ARG_STRING, 0, OPT_DAEMON, 0, 0 },
441 {"config", 0, POPT_ARG_STRING, 0, OPT_DAEMON, 0, 0 },
442 {"daemon", 0, POPT_ARG_NONE, 0, OPT_DAEMON, 0, 0 },
443 {"no-detach", 0, POPT_ARG_NONE, 0, OPT_DAEMON, 0, 0 },
3ac7f5d4
WD
444 {0,0,0,0, 0, 0, 0}
445};
446
447static void daemon_usage(enum logcode F)
448{
449 print_rsync_version(F);
450
451 rprintf(F,"\nUsage: rsync --daemon [OPTION]...\n");
452 rprintf(F," --address=ADDRESS bind to the specified address\n");
b3708acf 453 rprintf(F," --bwlimit=KBPS limit I/O bandwidth; KBytes per second\n");
3ac7f5d4
WD
454 rprintf(F," --config=FILE specify alternate rsyncd.conf file\n");
455 rprintf(F," --no-detach do not detach from the parent\n");
b3708acf 456 rprintf(F," --port=PORT listen on alternate port number\n");
1bd9db74 457 rprintf(F," -v, --verbose increase verbosity\n");
25ff0441 458#if INET6
3ac7f5d4
WD
459 rprintf(F," -4, --ipv4 prefer IPv4\n");
460 rprintf(F," -6, --ipv6 prefer IPv6\n");
461#endif
462 rprintf(F," -h, --help show this help screen\n");
463
464 rprintf(F,"\nIf you were not trying to invoke rsync as a daemon, avoid using any of the\n");
465 rprintf(F,"daemon-specific rsync options. See also the rsyncd.conf(5) man page.\n");
466}
467
468static struct poptOption long_daemon_options[] = {
469 /* longName, shortName, argInfo, argPtr, value, descrip, argDesc */
470 {"address", 0, POPT_ARG_STRING, &bind_address, 0, 0, 0 },
9fb08441 471 {"bwlimit", 0, POPT_ARG_INT, &daemon_bwlimit, 0, 0, 0 },
3ac7f5d4
WD
472 {"config", 0, POPT_ARG_STRING, &config_file, 0, 0, 0 },
473 {"daemon", 0, POPT_ARG_NONE, &daemon_opt, 0, 0, 0 },
25ff0441 474#if INET6
3ac7f5d4
WD
475 {"ipv4", '4', POPT_ARG_VAL, &default_af_hint, AF_INET, 0, 0 },
476 {"ipv6", '6', POPT_ARG_VAL, &default_af_hint, AF_INET6, 0, 0 },
477#endif
478 {"no-detach", 0, POPT_ARG_NONE, &no_detach, 0, 0, 0 },
479 {"port", 0, POPT_ARG_INT, &rsync_port, 0, 0, 0 },
480 {"protocol", 0, POPT_ARG_INT, &protocol_version, 0, 0, 0 },
481 {"server", 0, POPT_ARG_NONE, &am_server, 0, 0, 0 },
1bd9db74 482 {"verbose", 'v', POPT_ARG_NONE, 0, 'v', 0, 0 },
3ac7f5d4 483 {"help", 'h', POPT_ARG_NONE, 0, 'h', 0, 0 },
ad715008 484 {0,0,0,0, 0, 0, 0}
2855f61f 485};
7a6421fa 486
06963d0f 487
e51094b7 488static char err_buf[200];
cd8185f2 489
2855f61f 490
dafe63ca
MP
491/**
492 * Store the option error message, if any, so that we can log the
493 * connection attempt (which requires parsing the options), and then
494 * show the error later on.
495 **/
cd8185f2
AT
496void option_error(void)
497{
e51094b7
WD
498 if (!err_buf[0]) {
499 strcpy(err_buf, "Error parsing options: "
500 "option may be supported on client but not on server?\n");
cd8185f2 501 }
e51094b7 502
e51094b7 503 rprintf(FERROR, RSYNC_NAME ": %s", err_buf);
cd8185f2
AT
504}
505
dafe63ca
MP
506
507/**
27ed20f7
WD
508 * Tweak the option table to disable all options that the rsyncd.conf
509 * file has told us to refuse.
dafe63ca 510 **/
27ed20f7 511static void set_refuse_options(char *bp)
cd8185f2 512{
27ed20f7 513 struct poptOption *op;
093e816c 514 char *cp, shortname[2];
d73e7f6e 515 int is_wild, found_match;
093e816c
WD
516
517 shortname[1] = '\0';
27ed20f7
WD
518
519 while (1) {
06a50542
WD
520 while (*bp == ' ') bp++;
521 if (!*bp)
522 break;
27ed20f7
WD
523 if ((cp = strchr(bp, ' ')) != NULL)
524 *cp= '\0';
06a50542
WD
525 /* If they specify "delete", reject all delete options. */
526 if (strcmp(bp, "delete") == 0)
527 bp = "delete*";
093e816c 528 is_wild = strpbrk(bp, "*?[") != NULL;
d73e7f6e 529 found_match = 0;
55afbb52 530 for (op = long_options; ; op++) {
093e816c 531 *shortname = op->shortName;
d73e7f6e
WD
532 if (!op->longName && !*shortname)
533 break;
534 if ((op->longName && wildmatch(bp, op->longName))
535 || (*shortname && wildmatch(bp, shortname))) {
e57211c5
WD
536 if (op->argInfo == POPT_ARG_VAL)
537 op->argInfo = POPT_ARG_NONE;
06a50542 538 op->val = (op - long_options) + OPT_REFUSED_BASE;
d73e7f6e 539 found_match = 1;
06a50542
WD
540 if (!is_wild)
541 break;
27ed20f7 542 }
cd8185f2 543 }
d73e7f6e
WD
544 if (!found_match) {
545 rprintf(FLOG, "No match for refuse-options string \"%s\"\n",
546 bp);
547 }
27ed20f7
WD
548 if (!cp)
549 break;
550 *cp = ' ';
551 bp = cp + 1;
cd8185f2 552 }
cd8185f2
AT
553}
554
555
8db7cc2c 556static int count_args(const char **argv)
2855f61f 557{
dfa32483 558 int i = 0;
2855f61f 559
8db7cc2c
WD
560 if (argv) {
561 while (argv[i] != NULL)
562 i++;
563 }
dfa32483
WD
564
565 return i;
2855f61f
MP
566}
567
568
dafe63ca
MP
569/**
570 * Process command line arguments. Called on both local and remote.
571 *
572 * @retval 1 if all options are OK; with globals set to appropriate
573 * values
574 *
575 * @retval 0 on error, with err_buf containing an explanation
576 **/
2855f61f 577int parse_arguments(int *argc, const char ***argv, int frommain)
7a6421fa 578{
d853783f 579 int opt;
cd8185f2 580 char *ref = lp_refuse_options(module_id);
7be73df4 581 const char *arg;
dfa32483 582 poptContext pc;
d853783f 583
27ed20f7
WD
584 if (ref && *ref)
585 set_refuse_options(ref);
586
dfa32483 587 /* TODO: Call poptReadDefaultConfig; handle errors. */
cd8185f2 588
dfa32483
WD
589 /* The context leaks in case of an error, but if there's a
590 * problem we always exit anyhow. */
591 pc = poptGetContext(RSYNC_NAME, *argc, *argv, long_options, 0);
3359acb8
WD
592 {
593 struct poptAlias my_alias;
594 char **argv = new_array(char *, 1);
595 argv[0] = strdup("--delete-during");
596 my_alias.longName = "del", my_alias.shortName = '\0';
597 my_alias.argc = 1;
598 my_alias.argv = (const char **)argv;
599 if (argv && argv[0])
600 poptAddAlias(pc, my_alias, 0);
601 }
9fb08441 602 poptReadDefaultConfig(pc, 0);
2855f61f
MP
603
604 while ((opt = poptGetNextOpt(pc)) != -1) {
dfa32483
WD
605 /* most options are handled automatically by popt;
606 * only special cases are returned and listed here. */
2855f61f 607
d853783f
AT
608 switch (opt) {
609 case OPT_VERSION:
dfa32483 610 print_rsync_version(FINFO);
d853783f 611 exit_cleanup(0);
dfa32483 612
3ac7f5d4
WD
613 case OPT_DAEMON:
614 if (am_daemon) {
615 strcpy(err_buf, "Attempt to hack rsync thwarted!\n");
616 return 0;
617 }
618 poptFreeContext(pc);
619 pc = poptGetContext(RSYNC_NAME, *argc, *argv,
620 long_daemon_options, 0);
621 while ((opt = poptGetNextOpt(pc)) != -1) {
622 switch (opt) {
623 case 'h':
624 daemon_usage(FINFO);
625 exit_cleanup(0);
626
1bd9db74
WD
627 case 'v':
628 verbose++;
629 break;
630
3ac7f5d4
WD
631 default:
632 rprintf(FERROR,
633 "rsync: %s: %s (in daemon mode)\n",
634 poptBadOption(pc, POPT_BADOPTION_NOALIAS),
635 poptStrerror(opt));
636 goto daemon_error;
637 }
638 }
639 if (!daemon_opt) {
640 rprintf(FERROR, "Daemon option(s) used without --daemon.\n");
641 daemon_error:
642 rprintf(FERROR,
643 "(Type \"rsync --daemon --help\" for assistance with daemon mode.)\n");
644 exit_cleanup(RERR_SYNTAX);
645 }
646 *argv = poptGetArgs(pc);
647 *argc = count_args(*argv);
648 daemon_opt = 0;
649 am_daemon = 1;
650 return 1;
651
5b56cc19 652 case OPT_MODIFY_WINDOW:
dfa32483
WD
653 /* The value has already been set by popt, but
654 * we need to remember that we're using a
655 * non-default setting. */
5b56cc19
AT
656 modify_window_set = 1;
657 break;
1de50993 658
aa4d3b4c 659 case OPT_FILTER:
7842418b 660 add_filter(&filter_list, poptGetOptArg(pc), 0);
d853783f
AT
661 break;
662
aa4d3b4c 663 case OPT_EXCLUDE:
7842418b
WD
664 add_filter(&filter_list, poptGetOptArg(pc),
665 XFLG_DEF_EXCLUDE);
aa4d3b4c
WD
666 break;
667
d853783f 668 case OPT_INCLUDE:
7842418b
WD
669 add_filter(&filter_list, poptGetOptArg(pc),
670 XFLG_DEF_INCLUDE);
d853783f
AT
671 break;
672
673 case OPT_EXCLUDE_FROM:
93695764 674 case OPT_INCLUDE_FROM:
7be73df4 675 arg = poptGetOptArg(pc);
ba3db479
WD
676 if (sanitize_paths)
677 arg = sanitize_path(NULL, arg, NULL, 0);
7842418b 678 if (server_filter_list.head) {
ba3db479
WD
679 char *cp = (char *)arg;
680 clean_fname(cp, 1);
7842418b 681 if (check_filter(&server_filter_list, cp, 0) < 0)
ba3db479
WD
682 goto options_rejected;
683 }
7842418b 684 add_filter_file(&filter_list, arg, XFLG_FATAL_ERRORS
aa4d3b4c
WD
685 | (opt == OPT_INCLUDE_FROM ? XFLG_DEF_INCLUDE
686 : XFLG_DEF_EXCLUDE));
93695764
DD
687 break;
688
d853783f
AT
689 case 'h':
690 usage(FINFO);
691 exit_cleanup(0);
692
d853783f
AT
693 case 'v':
694 verbose++;
695 break;
7a6421fa 696
b86f0cef 697 case 'q':
f98c60bf
WD
698 if (frommain)
699 quiet++;
b86f0cef
DD
700 break;
701
d853783f
AT
702 case OPT_SENDER:
703 if (!am_server) {
704 usage(FERROR);
65417579 705 exit_cleanup(RERR_SYNTAX);
d853783f
AT
706 }
707 am_sender = 1;
708 break;
709
aa4d3b4c
WD
710 case 'F':
711 switch (++F_option_cnt) {
712 case 1:
7842418b 713 add_filter(&filter_list,
aa4d3b4c
WD
714 ": /.rsync-filter", 0);
715 break;
716 case 2:
7842418b 717 add_filter(&filter_list,
aa4d3b4c
WD
718 "- .rsync-filter", 0);
719 break;
720 }
721 break;
722
d9fcc198
AT
723 case 'P':
724 do_progress = 1;
725 keep_partial = 1;
726 break;
727
088aac85 728 case OPT_WRITE_BATCH:
9b3318b0 729 /* batch_name is already set */
088aac85
DD
730 write_batch = 1;
731 break;
732
76f79ba7 733 case OPT_READ_BATCH:
9b3318b0 734 /* batch_name is already set */
6902ed17
MP
735 read_batch = 1;
736 break;
ea5164d1 737
7d5acf1d 738 case OPT_MAX_SIZE:
cc80022e 739 for (arg = max_size_arg; isdigit(*(uchar*)arg); arg++) {}
7d5acf1d 740 if (*arg == '.')
cc80022e 741 for (arg++; isdigit(*(uchar*)arg); arg++) {}
7d5acf1d
WD
742 switch (*arg) {
743 case 'k': case 'K':
744 max_size = atof(max_size_arg) * 1024;
745 break;
746 case 'm': case 'M':
747 max_size = atof(max_size_arg) * 1024*1024;
748 break;
749 case 'g': case 'G':
750 max_size = atof(max_size_arg) * 1024*1024*1024;
751 break;
752 case '\0':
753 max_size = atof(max_size_arg);
754 break;
755 default:
756 max_size = 0;
757 break;
758 }
759 if (max_size <= 0) {
e012f858 760 snprintf(err_buf, sizeof err_buf,
7d5acf1d
WD
761 "--max-size value is invalid: %s\n",
762 max_size_arg);
e012f858 763 return 0;
7d5acf1d
WD
764 }
765 break;
766
99218d82
WD
767 case OPT_TIMEOUT:
768 if (io_timeout && io_timeout < select_timeout)
769 select_timeout = io_timeout;
770 break;
771
59c95e42
DD
772 case OPT_LINK_DEST:
773#if HAVE_LINK
59c95e42 774 link_dest = 1;
e012f858
WD
775 dest_option = "--link-dest";
776 goto set_dest_dir;
59c95e42 777#else
d175d7e1 778 snprintf(err_buf, sizeof err_buf,
dfa32483 779 "hard links are not supported on this %s\n",
59c95e42 780 am_server ? "server" : "client");
59c95e42
DD
781 return 0;
782#endif
783
e012f858
WD
784 case OPT_COPY_DEST:
785 copy_dest = 1;
786 dest_option = "--copy-dest";
787 goto set_dest_dir;
788
789 case OPT_COMPARE_DEST:
790 compare_dest = 1;
791 dest_option = "--compare-dest";
792 set_dest_dir:
3b26bba0 793 if (basis_dir_cnt >= MAX_BASIS_DIRS) {
e012f858
WD
794 snprintf(err_buf, sizeof err_buf,
795 "ERROR: at most %d %s args may be specified\n",
796 MAX_BASIS_DIRS, dest_option);
797 return 0;
798 }
799 arg = poptGetOptArg(pc);
800 if (sanitize_paths)
801 arg = sanitize_path(NULL, arg, NULL, 0);
802 basis_dir[basis_dir_cnt++] = (char *)arg;
803 break;
804
d853783f 805 default:
55afbb52 806 /* A large opt value means that set_refuse_options()
c67d1386
WD
807 * turned this option off (opt-BASE is its index). */
808 if (opt >= OPT_REFUSED_BASE) {
809 struct poptOption *op =
810 &long_options[opt-OPT_REFUSED_BASE];
27ed20f7 811 int n = snprintf(err_buf, sizeof err_buf,
a2570930 812 "The server is configured to refuse --%s\n",
27ed20f7
WD
813 op->longName) - 1;
814 if (op->shortName) {
815 snprintf(err_buf+n, sizeof err_buf-n,
816 " (-%c)\n", op->shortName);
817 }
818 } else {
8db7cc2c 819 snprintf(err_buf, sizeof err_buf, "%s%s: %s\n",
27ed20f7
WD
820 am_server ? "on remote machine: " : "",
821 poptBadOption(pc, POPT_BADOPTION_NOALIAS),
822 poptStrerror(opt));
823 }
dfa32483 824 return 0;
d853783f 825 }
7a6421fa 826 }
2855f61f 827
f1d5ba40
WD
828 if (am_sender < 0)
829 am_sender = 0;
830
e1add893 831#if !SUPPORT_LINKS
54e87b4b 832 if (preserve_links && !am_sender) {
e1add893
WD
833 snprintf(err_buf, sizeof err_buf,
834 "symlinks are not supported on this %s\n",
835 am_server ? "server" : "client");
e1add893
WD
836 return 0;
837 }
838#endif
839
840#if !SUPPORT_HARD_LINKS
841 if (preserve_hard_links) {
842 snprintf(err_buf, sizeof err_buf,
843 "hard links are not supported on this %s\n",
844 am_server ? "server" : "client");
e1add893
WD
845 return 0;
846 }
847#endif
848
088aac85 849 if (write_batch && read_batch) {
c3ea0990 850 snprintf(err_buf, sizeof err_buf,
36119893 851 "--write-batch and --read-batch can not be used together\n");
c3ea0990 852 return 0;
088aac85 853 }
94327ff0
WD
854 if (write_batch || read_batch) {
855 if (dry_run) {
c3ea0990 856 snprintf(err_buf, sizeof err_buf,
94327ff0
WD
857 "--%s-batch cannot be used with --dry_run (-n)\n",
858 write_batch ? "write" : "read");
c3ea0990 859 return 0;
94327ff0
WD
860 }
861 if (am_server) {
862 rprintf(FINFO,
863 "ignoring --%s-batch option sent to server\n",
864 write_batch ? "write" : "read");
865 /* We don't actually exit_cleanup(), so that we can
866 * still service older version clients that still send
867 * batch args to server. */
868 read_batch = write_batch = 0;
869 batch_name = NULL;
870 }
b9f592fb 871 }
36119893 872 if (read_batch && files_from) {
c3ea0990 873 snprintf(err_buf, sizeof err_buf,
36119893 874 "--read-batch cannot be used with --files-from\n");
c3ea0990 875 return 0;
36119893 876 }
9b3318b0 877 if (batch_name && strlen(batch_name) > MAX_BATCH_NAME_LEN) {
c3ea0990 878 snprintf(err_buf, sizeof err_buf,
9b3318b0
WD
879 "the batch-file name must be %d characters or less.\n",
880 MAX_BATCH_NAME_LEN);
c3ea0990 881 return 0;
beb93684 882 }
088aac85 883
ce58b1b4 884 if (tmpdir && strlen(tmpdir) >= MAXPATHLEN - 10) {
c3ea0990
WD
885 snprintf(err_buf, sizeof err_buf,
886 "the --temp-dir path is WAY too long.\n");
887 return 0;
ce58b1b4
WD
888 }
889
e012f858
WD
890 if (compare_dest + copy_dest + link_dest > 1) {
891 snprintf(err_buf, sizeof err_buf,
892 "You may not mix --compare-dest, --copy-dest, and --link-dest.\n");
893 return 0;
894 }
895
dfa32483 896 if (archive_mode) {
ea5164d1 897 if (!files_from)
51d48398 898 recurse = -1; /* infinite recursion */
dfa32483
WD
899#if SUPPORT_LINKS
900 preserve_links = 1;
901#endif
902 preserve_perms = 1;
903 preserve_times = 1;
904 preserve_gid = 1;
905 preserve_uid = 1;
906 preserve_devices = 1;
907 }
51d48398
WD
908
909 if (recurse || list_only || files_from)
65e4cda0 910 xfer_dirs |= 1;
dfa32483 911
ea5164d1
WD
912 if (relative_paths < 0)
913 relative_paths = files_from? 1 : 0;
914
c6eb7fad 915 if (!!delete_before + delete_during + delete_after > 1) {
3359acb8 916 snprintf(err_buf, sizeof err_buf,
c6eb7fad 917 "You may not combine multiple --delete-WHEN options.\n");
3359acb8
WD
918 return 0;
919 }
78fc60cd
WD
920 if (!recurse) {
921 delete_before = delete_during = delete_after = 0;
922 delete_mode = delete_excluded = 0;
923 } else if (delete_before || delete_during || delete_after)
51d48398 924 delete_mode = 1;
3359acb8
WD
925 else if (delete_mode || delete_excluded)
926 delete_mode = delete_before = 1;
51d48398 927
7be73df4 928 *argv = poptGetArgs(pc);
8db7cc2c 929 *argc = count_args(*argv);
7be73df4
WD
930
931 if (sanitize_paths) {
932 int i;
933 for (i = *argc; i-- > 0; )
10796f4b 934 (*argv)[i] = sanitize_path(NULL, (*argv)[i], "", 0);
7be73df4 935 if (tmpdir)
10796f4b 936 tmpdir = sanitize_path(NULL, tmpdir, NULL, 0);
a7260c40 937 if (partial_dir)
10796f4b 938 partial_dir = sanitize_path(NULL, partial_dir, NULL, 0);
7be73df4 939 if (backup_dir)
10796f4b 940 backup_dir = sanitize_path(NULL, backup_dir, NULL, 0);
7be73df4 941 if (files_from)
10796f4b 942 files_from = sanitize_path(NULL, files_from, NULL, 0);
7be73df4 943 }
7842418b
WD
944 if (server_filter_list.head && !am_sender) {
945 struct filter_list_struct *elp = &server_filter_list;
e012f858 946 int i;
c3ea0990 947 if (tmpdir) {
58b1999e 948 clean_fname(tmpdir, 1);
7842418b 949 if (check_filter(elp, tmpdir, 1) < 0)
c3ea0990
WD
950 goto options_rejected;
951 }
952 if (partial_dir) {
58b1999e 953 clean_fname(partial_dir, 1);
7842418b 954 if (check_filter(elp, partial_dir, 1) < 0)
c3ea0990
WD
955 goto options_rejected;
956 }
e012f858
WD
957 for (i = 0; i < basis_dir_cnt; i++) {
958 clean_fname(basis_dir[i], 1);
7842418b 959 if (check_filter(elp, basis_dir[i], 1) < 0)
c3ea0990
WD
960 goto options_rejected;
961 }
962 if (backup_dir) {
58b1999e 963 clean_fname(backup_dir, 1);
7842418b 964 if (check_filter(elp, backup_dir, 1) < 0)
c3ea0990
WD
965 goto options_rejected;
966 }
967 }
7842418b 968 if (server_filter_list.head && files_from) {
58b1999e 969 clean_fname(files_from, 1);
7842418b 970 if (check_filter(&server_filter_list, files_from, 0) < 0) {
c3ea0990
WD
971 options_rejected:
972 snprintf(err_buf, sizeof err_buf,
973 "Your options have been rejected by the server.\n");
974 return 0;
975 }
976 }
7be73df4 977
d175d7e1 978 if (!backup_suffix)
e0391f81 979 backup_suffix = backup_dir ? "" : BACKUP_SUFFIX;
d175d7e1 980 backup_suffix_len = strlen(backup_suffix);
80ddadb7 981 if (strchr(backup_suffix, '/') != NULL) {
c3ea0990
WD
982 snprintf(err_buf, sizeof err_buf,
983 "--suffix cannot contain slashes: %s\n",
80ddadb7 984 backup_suffix);
c3ea0990 985 return 0;
80ddadb7 986 }
e0391f81
WD
987 if (backup_dir) {
988 backup_dir_len = strlcpy(backup_dir_buf, backup_dir, sizeof backup_dir_buf);
989 backup_dir_remainder = sizeof backup_dir_buf - backup_dir_len;
990 if (backup_dir_remainder < 32) {
c3ea0990
WD
991 snprintf(err_buf, sizeof err_buf,
992 "the --backup-dir path is WAY too long.\n");
993 return 0;
e0391f81
WD
994 }
995 if (backup_dir_buf[backup_dir_len - 1] != '/') {
996 backup_dir_buf[backup_dir_len++] = '/';
997 backup_dir_buf[backup_dir_len] = '\0';
998 }
8dcf9335 999 if (verbose > 1 && !am_sender)
e0391f81 1000 rprintf(FINFO, "backup_dir is %s\n", backup_dir_buf);
8dcf9335 1001 } else if (!backup_suffix_len && (!am_server || !am_sender)) {
c3ea0990 1002 snprintf(err_buf, sizeof err_buf,
d175d7e1 1003 "--suffix cannot be a null string without --backup-dir\n");
c3ea0990 1004 return 0;
d175d7e1
WD
1005 }
1006
e2559dbe
S
1007 if (do_progress && !verbose)
1008 verbose = 1;
1009
9fb08441
WD
1010 if (daemon_bwlimit && (!bwlimit || bwlimit > daemon_bwlimit))
1011 bwlimit = daemon_bwlimit;
3c74c3a3
WD
1012 if (bwlimit) {
1013 bwlimit_writemax = (size_t)bwlimit * 128;
1014 if (bwlimit_writemax < 512)
1015 bwlimit_writemax = 512;
1016 }
1017
f06e7082
WD
1018 if (delay_updates && !partial_dir)
1019 partial_dir = ".~tmp~";
1020
a3221d2a
WD
1021 if (inplace) {
1022#if HAVE_FTRUNCATE
a7260c40
WD
1023 if (partial_dir) {
1024 snprintf(err_buf, sizeof err_buf,
f06e7082
WD
1025 "--inplace cannot be used with --%s\n",
1026 delay_updates ? "delay-updates" : "partial-dir");
a7260c40
WD
1027 return 0;
1028 }
a3221d2a 1029 keep_partial = 0;
ded4daf0
WD
1030#else
1031 snprintf(err_buf, sizeof err_buf,
1032 "--inplace is not supported on this %s\n",
1033 am_server ? "server" : "client");
1034 return 0;
1035#endif
075aa18f
WD
1036 } else {
1037 if (keep_partial && !partial_dir)
1038 partial_dir = getenv("RSYNC_PARTIAL_DIR");
1039 if (partial_dir) {
1040 if (!*partial_dir || strcmp(partial_dir, ".") == 0)
1041 partial_dir = NULL;
13791b1e 1042 else if (*partial_dir != '/') {
7842418b 1043 add_filter(&filter_list, partial_dir,
aa4d3b4c 1044 XFLG_DIRECTORY | XFLG_DEF_EXCLUDE);
13791b1e 1045 }
075aa18f
WD
1046 keep_partial = 1;
1047 }
a3221d2a
WD
1048 }
1049
ea5164d1
WD
1050 if (files_from) {
1051 char *colon;
c3ea0990 1052 if (*argc > 2 || (!am_daemon && *argc == 1)) {
ea5164d1
WD
1053 usage(FERROR);
1054 exit_cleanup(RERR_SYNTAX);
1055 }
63596e1c 1056 if (strcmp(files_from, "-") == 0) {
ea5164d1 1057 filesfrom_fd = 0;
63596e1c
WD
1058 if (am_server)
1059 remote_filesfrom_file = "-";
1060 }
ea5164d1
WD
1061 else if ((colon = find_colon(files_from)) != 0) {
1062 if (am_server) {
1063 usage(FERROR);
1064 exit_cleanup(RERR_SYNTAX);
1065 }
1066 remote_filesfrom_file = colon+1 + (colon[1] == ':');
1067 if (strcmp(remote_filesfrom_file, "-") == 0) {
c3ea0990
WD
1068 snprintf(err_buf, sizeof err_buf,
1069 "Invalid --files-from remote filename\n");
1070 return 0;
ea5164d1
WD
1071 }
1072 } else {
ea5164d1
WD
1073 filesfrom_fd = open(files_from, O_RDONLY|O_BINARY);
1074 if (filesfrom_fd < 0) {
c3ea0990
WD
1075 snprintf(err_buf, sizeof err_buf,
1076 "failed to open files-from file %s: %s\n",
1077 files_from, strerror(errno));
1078 return 0;
ea5164d1
WD
1079 }
1080 }
1081 }
1082
b11ed3b1 1083 return 1;
7a6421fa
AT
1084}
1085
1086
dafe63ca
MP
1087/**
1088 * Construct a filtered list of options to pass through from the
1089 * client to the server.
1090 *
1091 * This involves setting options that will tell the server how to
1092 * behave, and also filtering out options that are processed only
1093 * locally.
1094 **/
7a6421fa
AT
1095void server_options(char **args,int *argc)
1096{
e012f858 1097 static char argstr[50+MAX_BASIS_DIRS*2];
d853783f 1098 int ac = *argc;
f98c60bf 1099 char *arg;
ef5d23eb 1100
d853783f
AT
1101 int i, x;
1102
93689aa5
DD
1103 if (blocking_io == -1)
1104 blocking_io = 0;
1105
d853783f
AT
1106 args[ac++] = "--server";
1107
1312d9fc
WD
1108 if (daemon_over_rsh) {
1109 args[ac++] = "--daemon";
1110 *argc = ac;
1111 /* if we're passing --daemon, we're done */
1112 return;
1113 }
1114
d853783f
AT
1115 if (!am_sender)
1116 args[ac++] = "--sender";
1117
1118 x = 1;
1119 argstr[0] = '-';
f98c60bf 1120 for (i = 0; i < verbose; i++)
d853783f 1121 argstr[x++] = 'v';
f0b36a48 1122
b86f0cef 1123 /* the -q option is intentionally left out */
d853783f
AT
1124 if (make_backups)
1125 argstr[x++] = 'b';
1126 if (update_only)
1127 argstr[x++] = 'u';
1128 if (dry_run)
1129 argstr[x++] = 'n';
1130 if (preserve_links)
1131 argstr[x++] = 'l';
1132 if (copy_links)
1133 argstr[x++] = 'L';
65e4cda0 1134 if (xfer_dirs > 1)
5454d22a 1135 argstr[x++] = 'd';
716e73d4
WD
1136 if (keep_dirlinks && am_sender)
1137 argstr[x++] = 'K';
1bfbf40b 1138
dfa32483 1139 if (whole_file > 0)
d853783f 1140 argstr[x++] = 'W';
bceec82f
MP
1141 /* We don't need to send --no-whole-file, because it's the
1142 * default for remote transfers, and in any case old versions
1143 * of rsync will not understand it. */
dfa32483 1144
d853783f
AT
1145 if (preserve_hard_links)
1146 argstr[x++] = 'H';
1147 if (preserve_uid)
1148 argstr[x++] = 'o';
1149 if (preserve_gid)
1150 argstr[x++] = 'g';
1151 if (preserve_devices)
1152 argstr[x++] = 'D';
1153 if (preserve_times)
1154 argstr[x++] = 't';
20fb7b91
WD
1155 if (omit_dir_times && am_sender)
1156 argstr[x++] = 'O';
d853783f
AT
1157 if (preserve_perms)
1158 argstr[x++] = 'p';
51d48398 1159 if (recurse < 0)
d853783f
AT
1160 argstr[x++] = 'r';
1161 if (always_checksum)
1162 argstr[x++] = 'c';
1163 if (cvs_exclude)
1164 argstr[x++] = 'C';
1165 if (ignore_times)
1166 argstr[x++] = 'I';
1167 if (relative_paths)
1168 argstr[x++] = 'R';
1169 if (one_file_system)
1170 argstr[x++] = 'x';
1171 if (sparse_files)
1172 argstr[x++] = 'S';
1173 if (do_compression)
1174 argstr[x++] = 'z';
f0b36a48 1175
51d48398
WD
1176 /* This is a complete hack - blame Rusty. FIXME!
1177 * This hack is only needed for older rsync versions that
1178 * don't understand the --list-only option. */
1179 if (list_only == 1 && recurse >= 0)
f0b36a48
AT
1180 argstr[x++] = 'r';
1181
d853783f
AT
1182 argstr[x] = 0;
1183
f98c60bf
WD
1184 if (x != 1)
1185 args[ac++] = argstr;
d853783f 1186
51d48398
WD
1187 if (list_only > 1)
1188 args[ac++] = "--list-only";
1189
195bd906 1190 if (block_size) {
a06b419d 1191 if (asprintf(&arg, "-B%lu", block_size) < 0)
f98c60bf
WD
1192 goto oom;
1193 args[ac++] = arg;
dfa32483 1194 }
d853783f 1195
0b73ca12 1196 if (max_delete && am_sender) {
f98c60bf
WD
1197 if (asprintf(&arg, "--max-delete=%d", max_delete) < 0)
1198 goto oom;
1199 args[ac++] = arg;
dfa32483
WD
1200 }
1201
7d5acf1d
WD
1202 if (max_size && am_sender) {
1203 args[ac++] = "--max-size";
1204 args[ac++] = max_size_arg;
1205 }
1206
d853783f 1207 if (io_timeout) {
f98c60bf
WD
1208 if (asprintf(&arg, "--timeout=%d", io_timeout) < 0)
1209 goto oom;
1210 args[ac++] = arg;
dfa32483 1211 }
d853783f 1212
ef5d23eb 1213 if (bwlimit) {
f98c60bf
WD
1214 if (asprintf(&arg, "--bwlimit=%d", bwlimit) < 0)
1215 goto oom;
1216 args[ac++] = arg;
ef5d23eb
DD
1217 }
1218
d175d7e1
WD
1219 if (backup_dir) {
1220 args[ac++] = "--backup-dir";
1221 args[ac++] = backup_dir;
1222 }
1223
1224 /* Only send --suffix if it specifies a non-default value. */
f98c60bf 1225 if (strcmp(backup_suffix, backup_dir ? "" : BACKUP_SUFFIX) != 0) {
191e40da 1226 /* We use the following syntax to avoid weirdness with '~'. */
f98c60bf
WD
1227 if (asprintf(&arg, "--suffix=%s", backup_suffix) < 0)
1228 goto oom;
1229 args[ac++] = arg;
d853783f
AT
1230 }
1231
06a50542
WD
1232 if (am_sender) {
1233 if (delete_excluded)
1234 args[ac++] = "--delete-excluded";
c6eb7fad 1235 else if (delete_before == 1 || delete_after)
06a50542 1236 args[ac++] = "--delete";
57f74bd1 1237 if (delete_before > 1)
c6eb7fad 1238 args[ac++] = "--delete-before";
3359acb8
WD
1239 if (delete_during)
1240 args[ac++] = "--delete-during";
06a50542
WD
1241 if (delete_after)
1242 args[ac++] = "--delete-after";
06a50542
WD
1243 if (force_delete)
1244 args[ac++] = "--force";
1245 }
b33b791e 1246
f83f0548
AT
1247 if (size_only)
1248 args[ac++] = "--size-only";
1249
5b56cc19 1250 if (modify_window_set) {
f98c60bf
WD
1251 if (asprintf(&arg, "--modify-window=%d", modify_window) < 0)
1252 goto oom;
1253 args[ac++] = arg;
5b56cc19
AT
1254 }
1255
c8d895de 1256 if (checksum_seed) {
221ddb94 1257 if (asprintf(&arg, "--checksum-seed=%d", checksum_seed) < 0)
c8d895de
WD
1258 goto oom;
1259 args[ac++] = arg;
1260 }
1261
a7260c40
WD
1262 if (partial_dir && am_sender) {
1263 args[ac++] = "--partial-dir";
1264 args[ac++] = partial_dir;
f06e7082
WD
1265 if (delay_updates)
1266 args[ac++] = "--delay-updates";
a7260c40 1267 } else if (keep_partial)
d853783f
AT
1268 args[ac++] = "--partial";
1269
ef55c686
AT
1270 if (ignore_errors)
1271 args[ac++] = "--ignore-errors";
1272
b5313607
DD
1273 if (copy_unsafe_links)
1274 args[ac++] = "--copy-unsafe-links";
1275
d853783f
AT
1276 if (safe_symlinks)
1277 args[ac++] = "--safe-links";
1278
1279 if (numeric_ids)
1280 args[ac++] = "--numeric-ids";
1281
0b73ca12 1282 if (only_existing && am_sender)
1347d512
AT
1283 args[ac++] = "--existing";
1284
dfa32483 1285 if (opt_ignore_existing && am_sender)
3d6feada
MP
1286 args[ac++] = "--ignore-existing";
1287
a3221d2a
WD
1288 if (inplace)
1289 args[ac++] = "--inplace";
1290
d853783f
AT
1291 if (tmpdir) {
1292 args[ac++] = "--temp-dir";
1293 args[ac++] = tmpdir;
1294 }
1295
e012f858 1296 if (basis_dir[0] && am_sender) {
375a4556
DD
1297 /* the server only needs this option if it is not the sender,
1298 * and it may be an older version that doesn't know this
1299 * option, so don't send it if client is the sender.
1300 */
e012f858
WD
1301 int i;
1302 for (i = 0; i < basis_dir_cnt; i++) {
1303 args[ac++] = dest_option;
1304 args[ac++] = basis_dir[i];
1305 }
375a4556
DD
1306 }
1307
ea5164d1
WD
1308 if (files_from && (!am_sender || remote_filesfrom_file)) {
1309 if (remote_filesfrom_file) {
1310 args[ac++] = "--files-from";
1311 args[ac++] = remote_filesfrom_file;
1312 if (eol_nulls)
1313 args[ac++] = "--from0";
1314 } else {
1315 args[ac++] = "--files-from=-";
1316 args[ac++] = "--from0";
1317 }
c0ab28d1
WD
1318 if (!relative_paths)
1319 args[ac++] = "--no-relative";
ea5164d1 1320 }
c72f5bd9 1321 if (!implied_dirs && !am_sender)
3a90ea0a 1322 args[ac++] = "--no-implied-dirs";
ea5164d1 1323
d853783f 1324 *argc = ac;
f98c60bf
WD
1325 return;
1326
1327 oom:
1328 out_of_memory("server_options");
7a6421fa
AT
1329}
1330
ea5164d1
WD
1331/**
1332 * Return the position of a ':' IF it is not part of a filename (i.e. as
1333 * long as it doesn't occur after a slash.
1334 */
1335char *find_colon(char *s)
1336{
1337 char *p, *p2;
1338
1339 p = strchr(s,':');
f98c60bf
WD
1340 if (!p)
1341 return NULL;
ea5164d1
WD
1342
1343 /* now check to see if there is a / in the string before the : - if there is then
1344 discard the colon on the assumption that the : is part of a filename */
1345 p2 = strchr(s,'/');
f98c60bf
WD
1346 if (p2 && p2 < p)
1347 return NULL;
ea5164d1
WD
1348
1349 return p;
1350}