Improve RISC OS implementations of SDL_GetBasePath and SDL_GetPrefPath

main
Cameron Cawley 2021-09-22 14:01:00 +01:00 committed by Sam Lantinga
parent 3db898c5b6
commit 9ee6942e79
1 changed files with 144 additions and 26 deletions

View File

@ -25,28 +25,144 @@
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */ /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
/* System dependent filesystem routines */ /* System dependent filesystem routines */
#include <errno.h> #include <kernel.h>
#include <sys/stat.h> #include <swis.h>
#include <unixlib/local.h>
#include "SDL_error.h" #include "SDL_error.h"
#include "SDL_stdinc.h" #include "SDL_stdinc.h"
#include "SDL_filesystem.h" #include "SDL_filesystem.h"
#include "SDL_rwops.h"
/* Wrapper around __unixify_std that uses SDL's memory allocators */
static char *
SDL_unixify_std(const char *ro_path, char *buffer, size_t buf_len, int filetype)
{
const char *const in_buf = buffer; /* = NULL if we malloc the buffer. */
if (!buffer) {
/* This matches the logic in __unixify, with an additional byte for the
* extra path separator.
*/
buf_len = SDL_strlen(ro_path) + 14 + 1;
buffer = SDL_malloc(buf_len);
if (!buffer) {
SDL_OutOfMemory();
return NULL;
}
}
if (!__unixify_std(ro_path, buffer, buf_len, filetype)) {
if (!in_buf)
SDL_free(buffer);
SDL_SetError("Could not convert '%s' to a Unix-style path", ro_path);
return NULL;
}
/* HACK: It's necessary to add an extra path separator here since SDL's API
* requires it, however paths with trailing separators aren't normally valid
* on RISC OS.
*/
if (__get_riscosify_control() & __RISCOSIFY_NO_PROCESS)
SDL_strlcat(buffer, ".", buf_len);
else
SDL_strlcat(buffer, "/", buf_len);
return buffer;
}
static char *
canonicalisePath(const char *path, const char *pathVar)
{
_kernel_oserror *error;
_kernel_swi_regs regs;
char *buf;
regs.r[0] = 37;
regs.r[1] = (int)path;
regs.r[2] = 0;
regs.r[3] = (int)pathVar;
regs.r[4] = 0;
regs.r[5] = 0;
error = _kernel_swi(OS_FSControl, &regs, &regs);
if (error) {
SDL_SetError("Couldn't canonicalise path: %s", error->errmess);
return NULL;
}
regs.r[5] = 1 - regs.r[5];
buf = SDL_malloc(regs.r[5]);
if (!buf) {
SDL_OutOfMemory();
return NULL;
}
regs.r[2] = (int)buf;
error = _kernel_swi(OS_FSControl, &regs, &regs);
if (error) {
SDL_SetError("Couldn't canonicalise path: %s", error->errmess);
SDL_free(buf);
return NULL;
}
return buf;
}
static _kernel_oserror *
createDirectoryRecursive(char *path)
{
char *ptr = NULL;
_kernel_oserror *error;
_kernel_swi_regs regs;
regs.r[0] = 8;
regs.r[1] = (int)path;
regs.r[2] = 0;
for (ptr = path+1; *ptr; ptr++) {
if (*ptr == '.') {
*ptr = '\0';
error = _kernel_swi(OS_File, &regs, &regs);
*ptr = '.';
if (error != NULL)
return error;
}
}
return _kernel_swi(OS_File, &regs, &regs);
}
char * char *
SDL_GetBasePath(void) SDL_GetBasePath(void)
{ {
SDL_Unsupported(); _kernel_swi_regs regs;
return NULL; _kernel_oserror *error;
char *canon, *ptr, *retval;
error = _kernel_swi(OS_GetEnv, &regs, &regs);
if (error) {
return NULL;
}
canon = canonicalisePath((const char *)regs.r[0], "Run$Path");
if (!canon) {
return NULL;
}
/* chop off filename. */
ptr = SDL_strrchr(canon, '.');
if (ptr != NULL)
*ptr = '\0';
retval = SDL_unixify_std(canon, NULL, 0, __RISCOSIFY_FILETYPE_NOTSPECIFIED);
SDL_free(canon);
return retval;
} }
char * char *
SDL_GetPrefPath(const char *org, const char *app) SDL_GetPrefPath(const char *org, const char *app)
{ {
const char *prefix = "/<Choices$Write>/"; char *canon, *dir, *retval;
char *retval = NULL; size_t len;
char *ptr = NULL; _kernel_oserror *error;
size_t len = 0;
if (!app) { if (!app) {
SDL_InvalidParamError("app"); SDL_InvalidParamError("app");
@ -56,34 +172,36 @@ SDL_GetPrefPath(const char *org, const char *app)
org = ""; org = "";
} }
len = SDL_strlen(prefix) + SDL_strlen(org) + SDL_strlen(app) + 3; canon = canonicalisePath("<Choices$Write>", "Run$Path");
retval = (char *) SDL_malloc(len); if (!canon) {
if (!retval) { return NULL;
}
len = SDL_strlen(canon) + SDL_strlen(org) + SDL_strlen(app) + 4;
dir = (char *) SDL_malloc(len);
if (!dir) {
SDL_OutOfMemory(); SDL_OutOfMemory();
free(canon);
return NULL; return NULL;
} }
if (*org) { if (*org) {
SDL_snprintf(retval, len, "%s%s/%s/", prefix, org, app); SDL_snprintf(dir, len, "%s.%s.%s", canon, org, app);
} else { } else {
SDL_snprintf(retval, len, "%s%s/", prefix, app); SDL_snprintf(dir, len, "%s.%s", canon, app);
} }
for (ptr = retval+1; *ptr; ptr++) { SDL_free(canon);
if (*ptr == '/') {
*ptr = '\0'; error = createDirectoryRecursive(dir);
if (mkdir(retval, 0700) != 0 && errno != EEXIST) if (error != NULL) {
goto error; SDL_SetError("Couldn't create directory: %s", error->errmess);
*ptr = '/'; SDL_free(dir);
}
}
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 NULL;
} }
retval = SDL_unixify_std(dir, NULL, 0, __RISCOSIFY_FILETYPE_NOTSPECIFIED);
SDL_free(dir);
return retval; return retval;
} }