#include <sys/types.h>
#include <sys/stat.h>
#include <sys/socket.h>
#include <sys/un.h>
#include <errno.h>
#include <fcntl.h>
#include <limits.h>
#include <stdio.h>
#include <stdint.h>
#include <stdlib.h>
#include <string.h>
#include <ctype.h>
#include <unistd.h>
#include "extern.h"
enum tmpmode {
MKTEMP_NAME,
MKTEMP_FILE,
MKTEMP_DIR,
MKTEMP_LINK,
MKTEMP_FIFO,
MKTEMP_NOD,
MKTEMP_SOCK
};
#define TEMPCHARS "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789"
#define NUM_CHARS (sizeof(TEMPCHARS) - 1)
#define MIN_X 6
#define MKOTEMP_FLAGS (O_APPEND | O_CLOEXEC | O_DSYNC | O_RSYNC | O_SYNC)
#ifndef nitems
#define nitems(_a) (sizeof((_a)) / sizeof((_a)[0]))
#endif
static int
mktemp_internalat(int pfd, char *path, int slen, enum tmpmode mode,
int flags, const char *link, mode_t dev_type, dev_t dev)
{
char *start, *cp, *ep;
const char tempchars[] = TEMPCHARS;
unsigned int tries;
struct stat sb;
struct sockaddr_un sun;
size_t len;
int fd, saved_errno;
len = strlen(path);
if (len < MIN_X || slen < 0 || (size_t)slen > len - MIN_X) {
errno = EINVAL;
return(-1);
}
ep = path + len - slen;
for (start = ep; start > path && start[-1] == 'X'; start--)
;
if (ep - start < MIN_X) {
errno = EINVAL;
return(-1);
}
if (flags & ~MKOTEMP_FLAGS) {
errno = EINVAL;
return(-1);
}
flags |= O_CREAT | O_EXCL | O_RDWR;
tries = INT_MAX;
do {
cp = start;
do {
unsigned short rbuf[16];
unsigned int i;
arc4random_buf(rbuf, sizeof(rbuf));
for (i = 0; i < nitems(rbuf) && cp != ep; i++)
*cp++ = tempchars[rbuf[i] % NUM_CHARS];
} while (cp != ep);
switch (mode) {
case MKTEMP_NAME:
if (fstatat(pfd, path, &sb, AT_SYMLINK_NOFOLLOW) != 0)
return(errno == ENOENT ? 0 : -1);
break;
case MKTEMP_FILE:
fd = openat(pfd, path, flags, S_IRUSR|S_IWUSR);
if (fd != -1 || errno != EEXIST)
return(fd);
break;
case MKTEMP_DIR:
if (mkdirat(pfd, path, S_IRUSR|S_IWUSR|S_IXUSR) == 0)
return(0);
if (errno != EEXIST)
return(-1);
break;
case MKTEMP_LINK:
if (symlinkat(link, pfd, path) == 0)
return(0);
else if (errno != EEXIST)
return(-1);
break;
case MKTEMP_FIFO:
if (mkfifoat(pfd, path, S_IRUSR|S_IWUSR) == 0)
return(0);
else if (errno != EEXIST)
return(-1);
break;
case MKTEMP_NOD:
if (!(dev_type == S_IFCHR || dev_type == S_IFBLK)) {
errno = EINVAL;
return(-1);
}
if (mknodat(pfd, path, S_IRUSR|S_IWUSR|dev_type, dev)
== 0)
return(0);
else if (errno != EEXIST)
return(-1);
break;
case MKTEMP_SOCK:
memset(&sun, 0, sizeof(sun));
sun.sun_family = AF_UNIX;
if ((len = strlcpy(sun.sun_path, link,
sizeof(sun.sun_path))) >= sizeof(sun.sun_path)) {
errno = EINVAL;
return(-1);
}
if (sun.sun_path[len] != '/') {
if (strlcat(sun.sun_path, "/",
sizeof(sun.sun_path)) >=
sizeof(sun.sun_path)) {
errno = EINVAL;
return(-1);
}
}
if (strlcat(sun.sun_path, path, sizeof(sun.sun_path)) >=
sizeof(sun.sun_path)) {
errno = EINVAL;
return(-1);
}
if ((fd = socket(AF_UNIX, SOCK_STREAM | SOCK_CLOEXEC |
SOCK_NONBLOCK, 0)) == -1)
return -1;
if (bind(fd, (struct sockaddr *)&sun, sizeof(sun)) ==
0) {
close(fd);
return(0);
} else if (errno != EEXIST) {
saved_errno = errno;
close(fd);
errno = saved_errno;
return -1;
}
close(fd);
break;
}
} while (--tries);
errno = EEXIST;
return(-1);
}
int
mkstempat(int fd, char *path)
{
return mktemp_internalat(fd, path, 0, MKTEMP_FILE, 0, NULL, 0, 0);
}
char *
mkstemplinkat(char *link, int fd, char *path)
{
if (mktemp_internalat(fd, path, 0, MKTEMP_LINK, 0, link, 0, 0) == -1)
return NULL;
return path;
}
char *
mkstempfifoat(int fd, char *path)
{
if (mktemp_internalat(fd, path, 0, MKTEMP_FIFO, 0, NULL, 0, 0) == -1)
return NULL;
return path;
}
char *
mkstempnodat(int fd, char *path, mode_t mode, dev_t dev)
{
if (mktemp_internalat(fd, path, 0,
MKTEMP_NOD, 0, NULL, mode, dev) == -1)
return NULL;
return path;
}
char *
mkstempsock(const char *root, char *path)
{
if (mktemp_internalat(0, path, 0, MKTEMP_SOCK, 0, root, 0, 0) == -1)
return NULL;
return path;
}
int
mktemplate(char **ret, const char *path, int recursive)
{
int n, dirlen;
const char *cp;
if (recursive && (cp = strrchr(path, '/')) != NULL) {
dirlen = cp - path;
n = asprintf(ret, "%.*s/.%s.XXXXXXXXXX",
dirlen, path, path + dirlen + 1);
if (n == -1) {
ERR("asprintf");
*ret = NULL;
}
} else if ((n = asprintf(ret, ".%s.XXXXXXXXXX", path)) == -1) {
ERR("asprintf");
*ret = NULL;
}
return n;
}