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