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