X-Git-Url: http://nsz.repo.hu/git/?a=blobdiff_plain;f=src%2Fmisc%2Frealpath.c;h=88c849cda33de1e285a5b1eec7dfe9c222d013e5;hb=43e9f652bf4b2195b04fc14c93db591b30a7b790;hp=f6b55495781b4dbefaba09b4d33393ac476e0688;hpb=0b44a0315b47dd8eced9f3b7f31580cf14bbfc01;p=musl diff --git a/src/misc/realpath.c b/src/misc/realpath.c index f6b55495..88c849cd 100644 --- a/src/misc/realpath.c +++ b/src/misc/realpath.c @@ -1,6 +1,45 @@ #include +#include +#include +#include +#include +#include +#include +#include "syscall.h" -char *realpath(const char *filename, char *resolved) +void __procfdname(char *, unsigned); + +char *realpath(const char *restrict filename, char *restrict resolved) { + int fd; + ssize_t r; + struct stat st1, st2; + char buf[15+3*sizeof(int)]; + char tmp[PATH_MAX]; + + if (!filename) { + errno = EINVAL; + return 0; + } + + fd = sys_open(filename, O_PATH|O_NONBLOCK|O_CLOEXEC); + if (fd < 0) return 0; + __procfdname(buf, fd); + + r = readlink(buf, tmp, sizeof tmp - 1); + if (r < 0) goto err; + tmp[r] = 0; + + fstat(fd, &st1); + r = stat(tmp, &st2); + if (r<0 || st1.st_dev != st2.st_dev || st1.st_ino != st2.st_ino) { + if (!r) errno = ELOOP; + goto err; + } + + __syscall(SYS_close, fd); + return resolved ? strcpy(resolved, tmp) : strdup(tmp); +err: + __syscall(SYS_close, fd); return 0; }