From: Jim Meyering Date: Mon, 19 Sep 2005 15:45:05 +0000 (+0000) Subject: (unlinkat): New function. X-Git-Tag: v5.90~169 X-Git-Url: http://git.ipfire.org/?a=commitdiff_plain;h=ee8931ea8eae960a9f3224627a16a5e5c3ad7a82;p=thirdparty%2Fcoreutils.git (unlinkat): New function. --- diff --git a/lib/openat.c b/lib/openat.c index 8908931b2d..3e3251ba10 100644 --- a/lib/openat.c +++ b/lib/openat.c @@ -170,3 +170,43 @@ fstatat (int fd, char const *file, struct stat *st, int flag) errno = saved_errno; return err; } + +/* Replacement for Solaris' function by the same name. + + Simulate it by doing save_cwd/fchdir/(unlink|rmdir)/restore_cwd. + If either the save_cwd or the restore_cwd fails (relatively unlikely, + and usually indicative of a problem that deserves close attention), + then give a diagnostic and exit nonzero. + Otherwise, this function works just like Solaris' unlinkat. */ +int +unlinkat (int fd, char const *file, int flag) +{ + struct saved_cwd saved_cwd; + int saved_errno; + int err; + + if (fd == AT_FDCWD) + return (flag == AT_REMOVEDIR ? rmdir (file) : unlink (file)); + + if (save_cwd (&saved_cwd) != 0) + openat_save_fail (errno); + + if (fchdir (fd) != 0) + { + saved_errno = errno; + free_cwd (&saved_cwd); + errno = saved_errno; + return -1; + } + + err = (flag == AT_REMOVEDIR ? rmdir (file) : unlink (file)); + saved_errno = errno; + + if (restore_cwd (&saved_cwd) != 0) + openat_restore_fail (errno); + + free_cwd (&saved_cwd); + + errno = saved_errno; + return err; +}