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