summaryrefslogtreecommitdiffstats
path: root/lib/at.c
diff options
context:
space:
mode:
authorKarel Zak2011-05-18 15:55:15 +0200
committerKarel Zak2011-05-18 15:57:16 +0200
commitcdc9c4194aac3897feb2156ad2d3d5cb663b4365 (patch)
treee72ce8fa2db479aac0dadf827dddb7014cd90308 /lib/at.c
parentagetty: try next speed after CBREAK (diff)
downloadkernel-qcow2-util-linux-cdc9c4194aac3897feb2156ad2d3d5cb663b4365.tar.gz
kernel-qcow2-util-linux-cdc9c4194aac3897feb2156ad2d3d5cb663b4365.tar.xz
kernel-qcow2-util-linux-cdc9c4194aac3897feb2156ad2d3d5cb663b4365.zip
lib: [at.c] add readlink_at(), fix semantic for absolute paths
Signed-off-by: Karel Zak <kzak@redhat.com>
Diffstat (limited to 'lib/at.c')
-rw-r--r--lib/at.c53
1 files changed, 41 insertions, 12 deletions
diff --git a/lib/at.c b/lib/at.c
index e4838ec3c..6d1cd13f7 100644
--- a/lib/at.c
+++ b/lib/at.c
@@ -11,6 +11,8 @@
#include "at.h"
#include "c.h"
+#undef HAVE_FSTATAT
+
int fstat_at(int dir, const char *dirname, const char *filename,
struct stat *st, int nofollow)
{
@@ -18,14 +20,18 @@ int fstat_at(int dir, const char *dirname, const char *filename,
return fstatat(dir, filename, st,
nofollow ? AT_SYMLINK_NOFOLLOW : 0);
#else
- char path[PATH_MAX];
- int len;
+ if (*filename != '/') {
+ char path[PATH_MAX];
+ int len;
+
+ len = snprintf(path, sizeof(path), "%s/%s", dirname, filename);
+ if (len < 0 || len + 1 > sizeof(path))
+ return -1;
- len = snprintf(path, sizeof(path), "%s/%s", dirname, filename);
- if (len < 0 || len + 1 > sizeof(path))
- return -1;
+ return nofollow ? lstat(path, st) : stat(path, st);
+ }
- return nofollow ? lstat(path, st) : stat(path, st);
+ return nofollow ? lstat(filename, st) : stat(filename, st);
#endif
}
@@ -34,14 +40,17 @@ int open_at(int dir, const char *dirname, const char *filename, int flags)
#ifdef HAVE_FSTATAT
return openat(dir, filename, flags);
#else
- char path[PATH_MAX];
- int len;
+ if (*filename != '/') {
+ char path[PATH_MAX];
+ int len;
- len = snprintf(path, sizeof(path), "%s/%s", dirname, filename);
- if (len < 0 || len + 1 > sizeof(path))
- return -1;
+ len = snprintf(path, sizeof(path), "%s/%s", dirname, filename);
+ if (len < 0 || len + 1 > sizeof(path))
+ return -1;
- return open(path, flags);
+ return open(path, flags);
+ }
+ return open(filename, flags);
#endif
}
@@ -56,6 +65,26 @@ FILE *fopen_at(int dir, const char *dirname, const char *filename, int flags,
return fdopen(fd, mode);
}
+ssize_t readlink_at(int dir, const char *dirname, const char *pathname,
+ char *buf, size_t bufsiz)
+{
+#ifdef HAVE_FSTATAT
+ return readlinkat(dir, pathname, buf, bufsiz);
+#else
+ if (*pathname != '/') {
+ char path[PATH_MAX];
+ int len;
+
+ len = snprintf(path, sizeof(path), "%s/%s", dirname, pathname);
+ if (len < 0 || len + 1 > sizeof(path))
+ return -1;
+
+ return readlink(path, buf, bufsiz);
+ }
+ return readlink(pathname, buf, bufsiz);
+#endif
+}
+
#ifdef TEST_PROGRAM__DISABLED_FOR_NOW
#include <errno.h>
#include <sys/types.h>