/******************************************************************************* Copyright(C) Jonas 'Sortie' Termansen 2013, 2014, 2015. This file is part of the Sortix C Library. The Sortix C Library is free software: you can redistribute it and/or modify it under the terms of the GNU Lesser General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. The Sortix C Library is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. You should have received a copy of the GNU Lesser General Public License along with the Sortix C Library. If not, see . stdlib/mkostemps.c Make a unique temporary file path and open it. *******************************************************************************/ #include #include #include #include static const uint32_t NUM_CHARACTERS = 10 + 26 + 26; static inline char random_character(void) { uint32_t index = arc4random_uniform(NUM_CHARACTERS); if ( index < 10 ) return '0' + index; if ( index < 10 + 26 ) return 'a' + index - 10; if ( index < 10 + 26 + 26 ) return 'A' + index - (10 + 26); __builtin_unreachable(); } int mkostemps(char* templ, int suffixlen, int flags) { size_t templ_length = strlen(templ); if ( templ_length < 6 || suffixlen < 0 || templ_length - 6 < (size_t) suffixlen ) return errno = EINVAL, -1; size_t xpos = templ_length - (6 + suffixlen); for ( size_t i = 0; i < 6; i++ ) if ( templ[xpos + i] != 'X' ) return errno = EINVAL, -1; flags &= ~O_ACCMODE; do { for ( size_t i = 0; i < 6; i++ ) templ[xpos + i] = random_character(); int fd = open(templ, flags | O_RDWR | O_EXCL | O_CREAT, 0600); if ( 0 <= fd ) return fd; } while ( errno == EEXIST ); memcpy(templ + xpos, "XXXXXX", 6); return -1; }