/* * Copyright (c) 2024 Scott Duensing, scott@kangaroopunch.com * * Permission is hereby granted, free of charge, to any person obtaining a copy * of this software and associated documentation files (the "Software"), to deal * in the Software without restriction, including without limitation the rights * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell * copies of the Software, and to permit persons to whom the Software is * furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in * all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE * SOFTWARE. */ #include #include #include #include #include #include "util.h" char *utilCreateString(char *format, ...) { va_list args; char *string; va_start(args, format); string = utilCreateStringVArgs(format, args); va_end(args); return string; } __attribute__((__format__(__printf__, 1, 0))) char *utilCreateStringVArgs(char *format, va_list args) { va_list argsCopy; int32_t size = 0; char *buffer = NULL; va_copy(argsCopy, args); size = vsnprintf(NULL, 0, format, argsCopy) + 1; va_end(argsCopy); buffer = calloc(1, (size_t)size); if (buffer) { vsnprintf(buffer, (size_t)size, format, args); } return buffer; } void utilFixPathSeparators(char **path, bool slash) { int32_t i = 0; int32_t j = 0; char *work = *path; char *temp = NULL; // Flip path separators to whatever our OS wants & remove repeated separators. while (work[i] != 0) { // Correct separator if (work[i] == '\\' || work[i] == '/') { // Was the prior character a seprator? if (j == 0) { work[j++] = utilGetPathSeparator(); } else { if (work[j - 1] != utilGetPathSeparator()) { // No, accept it. work[j++] = utilGetPathSeparator(); } } } else { work[j++] = work[i]; } i++; } work[j] = 0; if (slash) { // Does this string end with a path separator? if (work[strlen(work) - 1] != utilGetPathSeparator()) { // No - append one. temp = strdup(work); free(work); work = malloc(sizeof(char) * (strlen(temp) + 2)); strcpy(work, temp); work[strlen(temp)] = utilGetPathSeparator(); work[strlen(temp) + 1] = 0; free(temp); } } *path = work; } char *utilGetLastPathComponent(char *pathname) { static char *start; int32_t x; start = pathname; // Scan through name and find the last path separator for (x=0; x<(int32_t)strlen(pathname); x++) { if (pathname[x] == '\\' || pathname[x] == '/') { start = &pathname[x + 1]; } } return strdup(start); } char utilGetPathSeparator(void) { #ifdef _WIN32 return '\\'; #else return '/'; #endif } char *utilGetUpToLastPathComponent(char *pathname) { static char *copy = NULL; bool dumb = false; // Using (copy == NULL) below didn't work after optimizations, so enter the dummy. int32_t x; x = (int32_t)(strlen(pathname) - strlen(utilGetLastPathComponent(pathname))) - 1; if (x < 0) x = 0; if (dumb) { free(copy); copy = NULL; } else { dumb = true; } copy = strdup(pathname); copy[x] = 0; utilFixPathSeparators(©, true); return copy; } bool utilMkDirP(const char *dir, const mode_t mode) { char tmp[UTIL_PATH_MAX]; char *p = NULL; struct stat sb; size_t len; // Make copy of dir. len = strnlen(dir, UTIL_PATH_MAX); if (len == 0 || len == UTIL_PATH_MAX) { return -1; } memcpy(tmp, dir, len); tmp[len] = '\0'; // Remove trailing slash. if (tmp[len - 1] == utilGetPathSeparator()) { tmp[len - 1] = '\0'; } // Does it already exist? if (stat(tmp, &sb) == 0) { if (S_ISDIR (sb.st_mode)) { return true; } } // Recursive mkdir. for (p = tmp + 1; *p; p++) { if (*p == utilGetPathSeparator()) { *p = 0; if (stat(tmp, &sb) != 0) { // Does not exist - create it. if (mkdir(tmp, mode) < 0) { return false; } } else { if (!S_ISDIR(sb.st_mode)) { // Not a directory return false; } } *p = utilGetPathSeparator(); } } // Check path if (stat(tmp, &sb) != 0) { // Does not exist - create it. if (mkdir(tmp, mode) < 0) { return false; } } else { if (!S_ISDIR(sb.st_mode)) { // Not a directory return false; } } return true; } char *utilReplaceExtension(char *org, char *new_ext) { char *ext; char *tmp = strdup(org); size_t newSize; char *newName; ext = strrchr(tmp , '.'); if (ext) *ext = 0; newSize = strlen(tmp) + strlen(new_ext) + 1; newName = malloc(newSize); sprintf(newName, "%s%s", tmp, new_ext); free(tmp); return newName; } int utilStricmp(char *a, char *b) { for (;; a++, b++) { int d = tolower((unsigned char)*a) - tolower((unsigned char)*b); if (d != 0 || !*a) { return d; } } }