123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170 |
- #include "defaults.h"
- #include <fcntl.h>
- #include <errno.h>
- #include <string.h>
- #ifdef _win32_
- #include "winint.h"
- #include <util/folder/dirut.h>
- #endif
- #include <util/random/random.h>
- #include "sysstat.h"
- static const unsigned char padchar[] =
- "0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz";
- static int
- GetTemp(char* path, int* doopen, int domkdir, int slen)
- {
- char *start, *trv, *suffp;
- char* pad;
- #ifndef _win32_
- struct stat sbuf;
- int rval;
- #endif
- ui32 rand;
- if (doopen != nullptr && domkdir) {
- errno = EINVAL;
- return (0);
- }
- trv = path;
- while (*trv != 0) {
- ++trv;
- }
- trv -= slen;
- suffp = trv;
- --trv;
- if (trv < path) {
- errno = EINVAL;
- return (0);
- }
-
- while (trv >= path && *trv == 'X') {
- rand = (RandomNumber<ui32>()) % (sizeof(padchar) - 1);
- *trv-- = padchar[rand];
- }
- start = trv + 1;
-
- if (doopen != nullptr || domkdir) {
- for (; trv > path; --trv) {
- if (*trv == '/') {
- *trv = '\0';
- #ifdef _win32_
- ui32 attr = ::GetFileAttributesA(path);
- *trv = '/';
- if (attr == 0xFFFFFFFF)
- return (0);
- if (!(attr & FILE_ATTRIBUTE_DIRECTORY)) {
- errno = ENOTDIR;
- return (0);
- }
- #else
- rval = stat(path, &sbuf);
- *trv = '/';
- if (rval != 0) {
- return (0);
- }
- if (!S_ISDIR(sbuf.st_mode)) {
- errno = ENOTDIR;
- return (0);
- }
- #endif
- break;
- }
- }
- }
- for (;;) {
- if (doopen) {
- if ((*doopen =
- open(path, O_CREAT | O_EXCL | O_RDWR, 0600)) >= 0) {
- return (1);
- }
- if (errno != EEXIST) {
- return (0);
- }
- } else if (domkdir) {
- if (Mkdir(path, S_IRWXU) == 0) {
- return (1);
- }
- if (errno != EEXIST) {
- return (0);
- }
- } else
- #ifdef _win32_
- if (::GetFileAttributesA(path) == INVALID_FILE_ATTRIBUTES)
- return (errno == ENOENT);
- #else
- if (lstat(path, &sbuf)) {
- return (errno == ENOENT);
- }
- #endif
-
- for (trv = start;;) {
- if (*trv == '\0' || trv == suffp) {
- return (0);
- }
- pad = strchr((char*)padchar, *trv);
- if (pad == nullptr || *++pad == '\0') {
- *trv++ = padchar[0];
- } else {
- *trv++ = *pad;
- break;
- }
- }
- }
-
- }
- extern "C" int mkstemps(char* path, int slen) {
- int fd;
- return (GetTemp(path, &fd, 0, slen) ? fd : -1);
- }
- #if defined(_win_)
- char* mkdtemp(char* path) {
- return (GetTemp(path, (int*)nullptr, 1, 0) ? path : (char*)nullptr);
- }
- #endif
|