/* sync - update the super block Copyright (C) 1994-2020 Free Software Foundation, Inc. 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 the Free Software Foundation, either version 3 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program. If not, see . */ /* Written by Jim Meyering */ #include #include #include #include #include #include "system.h" #include "die.h" #include "error.h" /* The official name of this program (e.g., no 'g' prefix). */ #define PROGRAM_NAME "sync" #define AUTHORS \ proper_name ("Jim Meyering"), \ proper_name ("Giuseppe Scrivano") #ifndef HAVE_SYNCFS # define HAVE_SYNCFS 0 #endif enum sync_mode { MODE_FILE, MODE_DATA, MODE_FILE_SYSTEM, MODE_SYNC }; static struct option const long_options[] = { {"data", no_argument, NULL, 'd'}, {"file-system", no_argument, NULL, 'f'}, {GETOPT_HELP_OPTION_DECL}, {GETOPT_VERSION_OPTION_DECL}, {NULL, 0, NULL, 0} }; void usage (int status) { if (status != EXIT_SUCCESS) emit_try_help (); else { printf (_("Usage: %s [OPTION] [FILE]...\n"), program_name); fputs (_("\ Synchronize cached writes to persistent storage\n\ \n\ If one or more files are specified, sync only them,\n\ or their containing file systems.\n\ \n\ "), stdout); fputs (_("\ -d, --data sync only file data, no unneeded metadata\n\ "), stdout); fputs (_("\ -f, --file-system sync the file systems that contain the files\n\ "), stdout); fputs (HELP_OPTION_DESCRIPTION, stdout); fputs (VERSION_OPTION_DESCRIPTION, stdout); emit_ancillary_info (PROGRAM_NAME); } exit (status); } /* Sync the specified FILE, or file systems associated with FILE. Return 1 on success. */ static bool sync_arg (enum sync_mode mode, char const *file) { bool ret = true; int open_flags = O_RDONLY | O_NONBLOCK; int fd; #ifdef _AIX /* AIX 7.1 fsync requires write access to file. */ if (mode == MODE_FILE) open_flags = O_WRONLY | O_NONBLOCK; #endif /* Note O_PATH might be supported with syncfs(), though as of Linux 3.18 is not. */ fd = open (file, open_flags); if (fd < 0) { /* Use the O_RDONLY errno, which is significant with directories for example. */ int rd_errno = errno; if (open_flags != (O_WRONLY | O_NONBLOCK)) fd = open (file, O_WRONLY | O_NONBLOCK); if (fd < 0) { error (0, rd_errno, _("error opening %s"), quoteaf (file)); return false; } } /* We used O_NONBLOCK above to not hang with fifos, so reset that here. */ int fdflags = fcntl (fd, F_GETFL); if (fdflags == -1 || fcntl (fd, F_SETFL, fdflags & ~O_NONBLOCK) < 0) { error (0, errno, _("couldn't reset non-blocking mode %s"), quoteaf (file)); ret = false; } if (ret == true) { int sync_status = -1; switch (mode) { case MODE_DATA: sync_status = fdatasync (fd); break; case MODE_FILE: sync_status = fsync (fd); break; #if HAVE_SYNCFS case MODE_FILE_SYSTEM: sync_status = syncfs (fd); break; #endif default: assert ("invalid sync_mode"); } if (sync_status < 0) { error (0, errno, _("error syncing %s"), quoteaf (file)); ret = false; } } if (close (fd) < 0) { error (0, errno, _("failed to close %s"), quoteaf (file)); ret = false; } return ret; } int main (int argc, char **argv) { int c; bool args_specified; bool arg_data = false, arg_file_system = false; enum sync_mode mode; bool ok = true; initialize_main (&argc, &argv); set_program_name (argv[0]); setlocale (LC_ALL, ""); bindtextdomain (PACKAGE, LOCALEDIR); textdomain (PACKAGE); atexit (close_stdout); while ((c = getopt_long (argc, argv, "df", long_options, NULL)) != -1) { switch (c) { case 'd': arg_data = true; break; case 'f': arg_file_system = true; break; case_GETOPT_HELP_CHAR; case_GETOPT_VERSION_CHAR (PROGRAM_NAME, AUTHORS); default: usage (EXIT_FAILURE); } } args_specified = optind < argc; if (arg_data && arg_file_system) { die (EXIT_FAILURE, 0, _("cannot specify both --data and --file-system")); } if (!args_specified && arg_data) die (EXIT_FAILURE, 0, _("--data needs at least one argument")); if (! args_specified || (arg_file_system && ! HAVE_SYNCFS)) mode = MODE_SYNC; else if (arg_file_system) mode = MODE_FILE_SYSTEM; else if (! arg_data) mode = MODE_FILE; else mode = MODE_DATA; if (mode == MODE_SYNC) sync (); else { for (; optind < argc; optind++) ok &= sync_arg (mode, argv[optind]); } return ok ? EXIT_SUCCESS : EXIT_FAILURE; }