--- embedaddon/rsync/pipe.c 2012/02/17 15:09:30 1.1.1.1 +++ embedaddon/rsync/pipe.c 2021/03/17 00:32:36 1.1.1.4 @@ -4,7 +4,7 @@ * Copyright (C) 1996-2000 Andrew Tridgell * Copyright (C) 1996 Paul Mackerras * Copyright (C) 2001, 2002 Martin Pool - * Copyright (C) 2004-2009 Wayne Davison + * Copyright (C) 2004-2020 Wayne Davison * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by @@ -26,10 +26,17 @@ extern int am_sender; extern int am_server; extern int blocking_io; extern int filesfrom_fd; -extern mode_t orig_umask; +extern int munge_symlinks; +extern int always_checksum; +extern int use_db; +extern char *db_config; extern char *logfile_name; +extern int remote_option_cnt; +extern const char **remote_options; extern struct chmod_mode_struct *chmod_modes; +int same_db = 0; + /** * Create a child connected to us via its stdin/stdout. * @@ -49,7 +56,7 @@ pid_t piped_child(char **command, int *f_in, int *f_ou int to_child_pipe[2]; int from_child_pipe[2]; - if (verbose >= 2) + if (DEBUG_GTE(CMD, 1)) print_child_argv("opening connection using:", command); if (fd_pair(to_child_pipe) < 0 || fd_pair(from_child_pipe) < 0) { @@ -64,10 +71,10 @@ pid_t piped_child(char **command, int *f_in, int *f_ou } if (pid == 0) { - if (dup2(to_child_pipe[0], STDIN_FILENO) < 0 || - close(to_child_pipe[1]) < 0 || - close(from_child_pipe[0]) < 0 || - dup2(from_child_pipe[1], STDOUT_FILENO) < 0) { + if (dup2(to_child_pipe[0], STDIN_FILENO) < 0 + || close(to_child_pipe[1]) < 0 + || close(from_child_pipe[0]) < 0 + || dup2(from_child_pipe[1], STDOUT_FILENO) < 0) { rsyserr(FERROR, errno, "Failed to dup/close"); exit_cleanup(RERR_IPC); } @@ -75,7 +82,6 @@ pid_t piped_child(char **command, int *f_in, int *f_ou close(to_child_pipe[0]); if (from_child_pipe[1] != STDOUT_FILENO) close(from_child_pipe[1]); - umask(orig_umask); set_blocking(STDIN_FILENO); if (blocking_io > 0) set_blocking(STDOUT_FILENO); @@ -115,8 +121,7 @@ pid_t local_child(int argc, char **argv, int *f_in, in /* The parent process is always the sender for a local rsync. */ assert(am_sender); - if (fd_pair(to_child_pipe) < 0 || - fd_pair(from_child_pipe) < 0) { + if (fd_pair(to_child_pipe) < 0 || fd_pair(from_child_pipe) < 0) { rsyserr(FERROR, errno, "pipe"); exit_cleanup(RERR_IPC); } @@ -131,6 +136,7 @@ pid_t local_child(int argc, char **argv, int *f_in, in am_sender = 0; am_server = 1; filesfrom_fd = -1; + munge_symlinks = 0; /* Each side needs its own option. */ chmod_modes = NULL; /* Let the sending side handle this. */ /* Let the client side handle this. */ @@ -139,10 +145,28 @@ pid_t local_child(int argc, char **argv, int *f_in, in logfile_close(); } - if (dup2(to_child_pipe[0], STDIN_FILENO) < 0 || - close(to_child_pipe[1]) < 0 || - close(from_child_pipe[0]) < 0 || - dup2(from_child_pipe[1], STDOUT_FILENO) < 0) { + if (remote_option_cnt) { + const char *db_config_save = db_config; + int rc = remote_option_cnt + 1; + const char **rv = remote_options; + if (!parse_arguments(&rc, &rv)) { + option_error(); + exit_cleanup(RERR_SYNTAX); + } + if (db_config == db_config_save) + same_db = db_config != NULL; + else if (!db_config || !db_config_save || strcmp(db_config, db_config_save) != 0) { + use_db = 0; + if (db_config) + db_read_config(FERROR, db_config); + } + } else if (use_db) + same_db = 1; + + if (dup2(to_child_pipe[0], STDIN_FILENO) < 0 + || close(to_child_pipe[1]) < 0 + || close(from_child_pipe[0]) < 0 + || dup2(from_child_pipe[1], STDOUT_FILENO) < 0) { rsyserr(FERROR, errno, "Failed to dup/close"); exit_cleanup(RERR_IPC); } @@ -156,14 +180,87 @@ pid_t local_child(int argc, char **argv, int *f_in, in child_main(argc, argv); } - if (close(from_child_pipe[1]) < 0 || - close(to_child_pipe[0]) < 0) { + if (close(from_child_pipe[1]) < 0 || close(to_child_pipe[0]) < 0) { rsyserr(FERROR, errno, "Failed to close"); exit_cleanup(RERR_IPC); } *f_in = from_child_pipe[0]; *f_out = to_child_pipe[1]; + + return pid; +} + +pid_t run_filter(char *command[], int out, int *pipe_to_filter) +{ + pid_t pid; + int pipefds[2]; + + if (DEBUG_GTE(CMD, 1)) + print_child_argv("opening connection using:", command); + + if (pipe(pipefds) < 0) { + rsyserr(FERROR, errno, "pipe"); + exit_cleanup(RERR_IPC); + } + + pid = do_fork(); + if (pid == -1) { + rsyserr(FERROR, errno, "fork"); + exit_cleanup(RERR_IPC); + } + + if (pid == 0) { + if (dup2(pipefds[0], STDIN_FILENO) < 0 + || close(pipefds[1]) < 0 + || dup2(out, STDOUT_FILENO) < 0) { + rsyserr(FERROR, errno, "Failed dup/close"); + exit_cleanup(RERR_IPC); + } + umask(orig_umask); + set_blocking(STDIN_FILENO); + if (blocking_io) + set_blocking(STDOUT_FILENO); + execvp(command[0], command); + rsyserr(FERROR, errno, "Failed to exec %s", command[0]); + exit_cleanup(RERR_IPC); + } + + if (close(pipefds[0]) < 0) { + rsyserr(FERROR, errno, "Failed to close"); + exit_cleanup(RERR_IPC); + } + + *pipe_to_filter = pipefds[1]; + + return pid; +} + +pid_t run_filter_on_file(char *command[], int out, int in) +{ + pid_t pid; + + if (DEBUG_GTE(CMD, 1)) + print_child_argv("opening connection using:", command); + + pid = do_fork(); + if (pid == -1) { + rsyserr(FERROR, errno, "fork"); + exit_cleanup(RERR_IPC); + } + + if (pid == 0) { + if (dup2(in, STDIN_FILENO) < 0 + || dup2(out, STDOUT_FILENO) < 0) { + rsyserr(FERROR, errno, "Failed to dup2"); + exit_cleanup(RERR_IPC); + } + if (blocking_io) + set_blocking(STDOUT_FILENO); + execvp(command[0], command); + rsyserr(FERROR, errno, "Failed to exec %s", command[0]); + exit_cleanup(RERR_IPC); + } return pid; }