blob: 3fb0d902160c2a5060e882a7ba1bb8d88a489e47 [file] [edit]
//===----------------------------------------------------------------------===//
//
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
// See https://llvm.org/LICENSE.txt for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===----------------------------------------------------------------------===//
///
/// \file
/// Implementation of mkstemp, a POSIX function that creates a unique temporary
/// file from a template string ending in at least six 'X' characters.
///
/// Replaces the trailing X's with random characters from the POSIX portable
/// filename character set, opens the file exclusively, and returns an open
/// file descriptor, retrying automatically on name collision. See:
/// https://pubs.opengroup.org/onlinepubs/9799919799/functions/mkdtemp.html
///
//===----------------------------------------------------------------------===//
#include "src/stdlib/mkstemp.h"
#include "hdr/errno_macros.h"
#include "hdr/fcntl_macros.h"
#include "src/__support/CPP/string_view.h"
#include "src/__support/OSUtil/linux/syscall_wrappers/getrandom.h"
#include "src/__support/OSUtil/linux/syscall_wrappers/open.h"
#include "src/__support/common.h"
#include "src/__support/libc_errno.h"
#include "src/__support/macros/config.h"
#include "src/__support/macros/null_check.h"
namespace LIBC_NAMESPACE_DECL {
LLVM_LIBC_FUNCTION(int, mkstemp, (char *tmpl)) {
LIBC_CRASH_ON_NULLPTR(tmpl);
cpp::string_view str_view(tmpl);
size_t count = 0;
size_t len = str_view.size();
for (size_t i = len; i > 0; i--) {
if (str_view[i - 1] != 'X')
break;
count++;
}
if (count < 6) {
libc_errno = EINVAL;
return -1;
}
char *suffix = tmpl + len - count;
// POSIX portable filename character set, sorted by ASCII value.
// See
// https://pubs.opengroup.org/onlinepubs/9799919799/basedefs/V1_chap03.html#tag_03_265
const char charset[] = "-._0123456789"
"ABCDEFGHIJKLMNOPQRSTUVWXYZ"
"abcdefghijklmnopqrstuvwxyz";
int result = -1;
bool file_created = false;
while (!file_created) {
for (size_t i = 0; i < count; i++) {
uint8_t rand_byte;
auto ret = linux_syscalls::getrandom(&rand_byte, 1, 0);
if (!ret.has_value()) {
libc_errno = ret.error();
return -1;
}
// sizeof(charset) - 1 to account for the null terminator
suffix[i] = charset[rand_byte % (sizeof(charset) - 1)];
}
auto fd = linux_syscalls::open(tmpl, O_RDWR | O_CREAT | O_EXCL, 0600);
if (!fd.has_value()) {
if (fd.error() == EEXIST)
continue;
libc_errno = fd.error();
return -1;
}
result = fd.value();
file_created = true;
}
return result;
}
} // namespace LIBC_NAMESPACE_DECL