123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337 |
- /*
- Simple DirectMedia Layer
- Copyright (C) 1997-2022 Sam Lantinga <[email protected]>
- This software is provided 'as-is', without any express or implied
- warranty. In no event will the authors be held liable for any damages
- arising from the use of this software.
- Permission is granted to anyone to use this software for any purpose,
- including commercial applications, and to alter it and redistribute it
- freely, subject to the following restrictions:
- 1. The origin of this software must not be misrepresented; you must not
- claim that you wrote the original software. If you use this software
- in a product, an acknowledgment in the product documentation would be
- appreciated but is not required.
- 2. Altered source versions must be plainly marked as such, and must not be
- misrepresented as being the original software.
- 3. This notice may not be removed or altered from any source distribution.
- */
- #include "../../SDL_internal.h"
- #ifdef SDL_FILESYSTEM_UNIX
- /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
- /* System dependent filesystem routines */
- #include <sys/stat.h>
- #include <sys/types.h>
- #include <errno.h>
- #include <fcntl.h>
- #include <limits.h>
- #include <stdlib.h>
- #include <string.h>
- #include <unistd.h>
- #if defined(__FREEBSD__) || defined(__OPENBSD__)
- #include <sys/sysctl.h>
- #endif
- static char *
- readSymLink(const char *path)
- {
- char *retval = NULL;
- ssize_t len = 64;
- ssize_t rc = -1;
- while (1) {
- char *ptr = (char *) SDL_realloc(retval, (size_t) len);
- if (ptr == NULL) {
- SDL_OutOfMemory();
- break;
- }
- retval = ptr;
- rc = readlink(path, retval, len);
- if (rc == -1) {
- break; /* not a symlink, i/o error, etc. */
- } else if (rc < len) {
- retval[rc] = '\0'; /* readlink doesn't null-terminate. */
- return retval; /* we're good to go. */
- }
- len *= 2; /* grow buffer, try again. */
- }
- SDL_free(retval);
- return NULL;
- }
- #if defined(__OPENBSD__)
- static char *search_path_for_binary(const char *bin)
- {
- char *envr = SDL_getenv("PATH");
- size_t alloc_size;
- char *exe = NULL;
- char *start = envr;
- char *ptr;
- if (envr == NULL) {
- SDL_SetError("No $PATH set");
- return NULL;
- }
- envr = SDL_strdup(envr);
- if (envr == NULL) {
- SDL_OutOfMemory();
- return NULL;
- }
- SDL_assert(bin != NULL);
- alloc_size = SDL_strlen(bin) + SDL_strlen(envr) + 2;
- exe = (char *) SDL_malloc(alloc_size);
- do {
- ptr = SDL_strchr(start, ':'); /* find next $PATH separator. */
- if (ptr != start) {
- if (ptr) {
- *ptr = '\0';
- }
- /* build full binary path... */
- SDL_snprintf(exe, alloc_size, "%s%s%s", start, (ptr && (ptr[-1] == '/')) ? "" : "/", bin);
- if (access(exe, X_OK) == 0) { /* Exists as executable? We're done. */
- SDL_free(envr);
- return exe;
- }
- }
- start = ptr + 1; /* start points to beginning of next element. */
- } while (ptr != NULL);
- SDL_free(envr);
- SDL_free(exe);
- SDL_SetError("Process not found in $PATH");
- return NULL; /* doesn't exist in path. */
- }
- #endif
- char *
- SDL_GetBasePath(void)
- {
- char *retval = NULL;
- #if defined(__FREEBSD__)
- char fullpath[PATH_MAX];
- size_t buflen = sizeof (fullpath);
- const int mib[] = { CTL_KERN, KERN_PROC, KERN_PROC_PATHNAME, -1 };
- if (sysctl(mib, SDL_arraysize(mib), fullpath, &buflen, NULL, 0) != -1) {
- retval = SDL_strdup(fullpath);
- if (retval == NULL) {
- SDL_OutOfMemory();
- return NULL;
- }
- }
- #endif
- #if defined(__OPENBSD__)
- /* Please note that this will fail if the process was launched with a relative path and $PWD + the cwd have changed, or argv is altered. So don't do that. Or add a new sysctl to OpenBSD. */
- char **cmdline;
- size_t len;
- const int mib[] = { CTL_KERN, KERN_PROC_ARGS, getpid(), KERN_PROC_ARGV };
- if (sysctl(mib, 4, NULL, &len, NULL, 0) != -1) {
- char *exe, *pwddst;
- char *realpathbuf = (char *) SDL_malloc(PATH_MAX + 1);
- if (realpathbuf == NULL) {
- SDL_OutOfMemory();
- return NULL;
- }
- cmdline = SDL_malloc(len);
- if (cmdline == NULL) {
- SDL_free(realpathbuf);
- SDL_OutOfMemory();
- return NULL;
- }
- sysctl(mib, 4, cmdline, &len, NULL, 0);
- exe = cmdline[0];
- pwddst = NULL;
- if (SDL_strchr(exe, '/') == NULL) { /* not a relative or absolute path, check $PATH for it */
- exe = search_path_for_binary(cmdline[0]);
- } else {
- if (exe && *exe == '.') {
- const char *pwd = SDL_getenv("PWD");
- if (pwd && *pwd) {
- SDL_asprintf(&pwddst, "%s/%s", pwd, exe);
- }
- }
- }
- if (exe) {
- if (pwddst == NULL) {
- if (realpath(exe, realpathbuf) != NULL) {
- retval = realpathbuf;
- }
- } else {
- if (realpath(pwddst, realpathbuf) != NULL) {
- retval = realpathbuf;
- }
- SDL_free(pwddst);
- }
- if (exe != cmdline[0]) {
- SDL_free(exe);
- }
- }
- if (retval == NULL) {
- SDL_free(realpathbuf);
- }
- SDL_free(cmdline);
- }
- #endif
- #if defined(__SOLARIS__)
- const char *path = getexecname();
- if ((path != NULL) && (path[0] == '/')) { /* must be absolute path... */
- retval = SDL_strdup(path);
- if (retval == NULL) {
- SDL_OutOfMemory();
- return NULL;
- }
- }
- #endif
- /* is a Linux-style /proc filesystem available? */
- if (retval == NULL && (access("/proc", F_OK) == 0)) {
- /* !!! FIXME: after 2.0.6 ships, let's delete this code and just
- use the /proc/%llu version. There's no reason to have
- two copies of this plus all the #ifdefs. --ryan. */
- #if defined(__FREEBSD__)
- retval = readSymLink("/proc/curproc/file");
- #elif defined(__NETBSD__)
- retval = readSymLink("/proc/curproc/exe");
- #else
- retval = readSymLink("/proc/self/exe"); /* linux. */
- if (retval == NULL) {
- /* older kernels don't have /proc/self ... try PID version... */
- char path[64];
- const int rc = SDL_snprintf(path, sizeof(path),
- "/proc/%llu/exe",
- (unsigned long long) getpid());
- if ( (rc > 0) && (rc < sizeof(path)) ) {
- retval = readSymLink(path);
- }
- }
- #endif
- }
- /* If we had access to argv[0] here, we could check it for a path,
- or troll through $PATH looking for it, too. */
- if (retval != NULL) { /* chop off filename. */
- char *ptr = SDL_strrchr(retval, '/');
- if (ptr != NULL) {
- *(ptr+1) = '\0';
- } else { /* shouldn't happen, but just in case... */
- SDL_free(retval);
- retval = NULL;
- }
- }
- if (retval != NULL) {
- /* try to shrink buffer... */
- char *ptr = (char *) SDL_realloc(retval, SDL_strlen(retval) + 1);
- if (ptr != NULL) {
- retval = ptr; /* oh well if it failed. */
- }
- }
- return retval;
- }
- char *
- SDL_GetPrefPath(const char *org, const char *app)
- {
- /*
- * We use XDG's base directory spec, even if you're not on Linux.
- * This isn't strictly correct, but the results are relatively sane
- * in any case.
- *
- * http://standards.freedesktop.org/basedir-spec/basedir-spec-latest.html
- */
- const char *envr = SDL_getenv("XDG_DATA_HOME");
- const char *append;
- char *retval = NULL;
- char *ptr = NULL;
- size_t len = 0;
- if (app == NULL) {
- SDL_InvalidParamError("app");
- return NULL;
- }
- if (org == NULL) {
- org = "";
- }
- if (envr == NULL) {
- /* You end up with "$HOME/.local/share/Game Name 2" */
- envr = SDL_getenv("HOME");
- if (envr == NULL) {
- /* we could take heroic measures with /etc/passwd, but oh well. */
- SDL_SetError("neither XDG_DATA_HOME nor HOME environment is set");
- return NULL;
- }
- append = "/.local/share/";
- } else {
- append = "/";
- }
- len = SDL_strlen(envr);
- if (envr[len - 1] == '/') {
- append += 1;
- }
- len += SDL_strlen(append) + SDL_strlen(org) + SDL_strlen(app) + 3;
- retval = (char *) SDL_malloc(len);
- if (retval == NULL) {
- SDL_OutOfMemory();
- return NULL;
- }
- if (*org) {
- SDL_snprintf(retval, len, "%s%s%s/%s/", envr, append, org, app);
- } else {
- SDL_snprintf(retval, len, "%s%s%s/", envr, append, app);
- }
- for (ptr = retval+1; *ptr; ptr++) {
- if (*ptr == '/') {
- *ptr = '\0';
- if (mkdir(retval, 0700) != 0 && errno != EEXIST) {
- goto error;
- }
- *ptr = '/';
- }
- }
- if (mkdir(retval, 0700) != 0 && errno != EEXIST) {
- error:
- SDL_SetError("Couldn't create directory '%s': '%s'", retval, strerror(errno));
- SDL_free(retval);
- return NULL;
- }
- return retval;
- }
- #endif /* SDL_FILESYSTEM_UNIX */
- /* vi: set ts=4 sw=4 expandtab: */
|