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