mirror of
https://github.com/Ryujinx/SDL.git
synced 2025-03-27 10:35:13 +00:00
251 lines
7.1 KiB
C
251 lines
7.1 KiB
C
/*
|
|
Simple DirectMedia Layer
|
|
Copyright (C) 1997-2019 Sam Lantinga <slouken@libsdl.org>
|
|
|
|
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 <errno.h>
|
|
#include <stdio.h>
|
|
#include <unistd.h>
|
|
#include <stdlib.h>
|
|
#include <sys/stat.h>
|
|
#include <sys/types.h>
|
|
#include <limits.h>
|
|
#include <fcntl.h>
|
|
|
|
#if defined(__FREEBSD__) || defined(__OPENBSD__)
|
|
#include <sys/sysctl.h>
|
|
#endif
|
|
|
|
#include "SDL_error.h"
|
|
#include "SDL_stdinc.h"
|
|
#include "SDL_filesystem.h"
|
|
#include "SDL_rwops.h"
|
|
|
|
/* QNX's /proc/self/exefile is a text file and not a symlink. */
|
|
#if !defined(__QNXNTO__)
|
|
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;
|
|
}
|
|
#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) {
|
|
SDL_OutOfMemory();
|
|
return NULL;
|
|
}
|
|
}
|
|
#endif
|
|
#if defined(__OPENBSD__)
|
|
char **retvalargs;
|
|
size_t len;
|
|
const int mib[] = { CTL_KERN, KERN_PROC_ARGS, getpid(), KERN_PROC_ARGV };
|
|
if (sysctl(mib, 4, NULL, &len, NULL, 0) != -1) {
|
|
retvalargs = SDL_malloc(len);
|
|
if (!retvalargs) {
|
|
SDL_OutOfMemory();
|
|
return NULL;
|
|
}
|
|
sysctl(mib, 4, retvalargs, &len, NULL, 0);
|
|
retval = SDL_malloc(PATH_MAX + 1);
|
|
if (retval)
|
|
realpath(retvalargs[0], retval);
|
|
|
|
SDL_free(retvalargs);
|
|
}
|
|
#endif
|
|
#if defined(__SOLARIS__)
|
|
const char *path = getexecname();
|
|
if ((path != NULL) && (path[0] == '/')) { /* must be absolute path... */
|
|
retval = SDL_strdup(path);
|
|
if (!retval) {
|
|
SDL_OutOfMemory();
|
|
return NULL;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
/* is a Linux-style /proc filesystem available? */
|
|
if (!retval && (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");
|
|
#elif defined(__QNXNTO__)
|
|
retval = SDL_LoadFile("/proc/self/exefile", NULL);
|
|
#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, 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) {
|
|
SDL_InvalidParamError("app");
|
|
return NULL;
|
|
}
|
|
if (!org) {
|
|
org = "";
|
|
}
|
|
|
|
if (!envr) {
|
|
/* You end up with "$HOME/.local/share/Game Name 2" */
|
|
envr = SDL_getenv("HOME");
|
|
if (!envr) {
|
|
/* 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) {
|
|
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: */
|