/* listfile.c -- run a function in a specific directory Copyright (C) 2007, 2008, 2009, 2010, 2011 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 . */ /* This file was written by James Youngman, based on gnulib'c at-func.c. */ /* config.h must be included first. */ #include /* system headers. */ #include #include #include #include /* gnulib headers. */ #include "fcntl--.h" #include "openat.h" #include "save-cwd.h" /* find headers. */ #include "dircallback.h" int run_in_dir (const struct saved_cwd *there, int (*callback)(void*), void *usercontext) { int err = -1; int saved_errno = 0; struct saved_cwd here; if (0 == save_cwd (&here)) { if (0 == restore_cwd (there)) { err = callback(usercontext); saved_errno = (err < 0 ? errno : 0); } else { openat_restore_fail (errno); } if (restore_cwd (&here) != 0) openat_restore_fail (errno); free_cwd (&here); } else { openat_save_fail (errno); } if (saved_errno) errno = saved_errno; return err; } int run_in_dirfd (int dir_fd, int (*callback)(void*), void *usercontext) { if (dir_fd == AT_FDCWD) { return (*callback)(usercontext); } else { struct saved_cwd saved_cwd; int saved_errno; int err; if (save_cwd (&saved_cwd) != 0) openat_save_fail (errno); if (fchdir (dir_fd) != 0) { saved_errno = errno; free_cwd (&saved_cwd); errno = saved_errno; return -1; } err = (*callback)(usercontext); saved_errno = (err < 0 ? errno : 0); if (restore_cwd (&saved_cwd) != 0) openat_restore_fail (errno); free_cwd (&saved_cwd); if (saved_errno) errno = saved_errno; return err; } }