2006-04-23 06:44:19 +00:00
|
|
|
/*
|
2019-02-20 21:30:11 +00:00
|
|
|
* Playing-field leveler for Build
|
2006-04-23 06:44:19 +00:00
|
|
|
*/
|
|
|
|
|
2017-06-14 06:59:50 +00:00
|
|
|
#define LIBDIVIDE_BODY
|
2014-11-24 06:30:47 +00:00
|
|
|
#include "compat.h"
|
2019-10-20 08:52:38 +00:00
|
|
|
#include "debugbreak.h"
|
2014-11-24 06:30:47 +00:00
|
|
|
|
2006-04-23 06:44:19 +00:00
|
|
|
#ifdef _WIN32
|
2017-02-25 08:15:01 +00:00
|
|
|
# define NEED_SHLOBJ_H
|
|
|
|
# include "windows_inc.h"
|
2014-12-08 04:31:57 +00:00
|
|
|
#elif __APPLE__
|
|
|
|
# include "osxbits.h"
|
2006-04-23 06:44:19 +00:00
|
|
|
#endif
|
|
|
|
|
2019-03-01 08:51:50 +00:00
|
|
|
#ifndef USE_PHYSFS
|
2009-07-09 02:29:48 +00:00
|
|
|
#if defined(_MSC_VER)
|
2006-04-23 06:44:19 +00:00
|
|
|
# include <io.h>
|
|
|
|
#else
|
|
|
|
# include <dirent.h>
|
|
|
|
#endif
|
2019-03-01 08:51:50 +00:00
|
|
|
#endif
|
2006-04-23 06:44:19 +00:00
|
|
|
|
2015-03-02 07:54:24 +00:00
|
|
|
#if defined __linux || defined EDUKE32_BSD
|
2014-12-08 04:31:57 +00:00
|
|
|
# include <libgen.h> // for dirname()
|
|
|
|
#endif
|
2016-06-06 22:13:05 +00:00
|
|
|
#if defined EDUKE32_BSD
|
2015-03-02 07:54:24 +00:00
|
|
|
# include <limits.h> // for PATH_MAX
|
2014-12-08 04:31:57 +00:00
|
|
|
# include <sys/sysctl.h> // for sysctl() to get path to executable
|
|
|
|
#endif
|
|
|
|
|
2010-01-16 20:17:33 +00:00
|
|
|
#include "baselayer.h"
|
2006-04-23 06:44:19 +00:00
|
|
|
|
2014-05-30 00:02:16 +00:00
|
|
|
////////// PANICKING ALLOCATION FUNCTIONS //////////
|
|
|
|
|
|
|
|
static void (*g_MemErrHandler)(int32_t line, const char *file, const char *func);
|
|
|
|
|
|
|
|
#ifdef DEBUGGINGAIDS
|
|
|
|
static const char *g_MemErrFunc = "???";
|
|
|
|
static const char *g_MemErrFile = "???";
|
|
|
|
static int32_t g_MemErrLine;
|
|
|
|
|
|
|
|
void xalloc_set_location(int32_t line, const char *file, const char *func)
|
|
|
|
{
|
|
|
|
g_MemErrLine = line;
|
|
|
|
g_MemErrFile = file;
|
|
|
|
|
|
|
|
if (func)
|
|
|
|
g_MemErrFunc = func;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2018-10-25 23:30:22 +00:00
|
|
|
void *handle_memerr(void *p)
|
2014-05-30 00:02:16 +00:00
|
|
|
{
|
2018-10-25 23:30:22 +00:00
|
|
|
UNREFERENCED_PARAMETER(p);
|
2018-10-07 05:21:20 +00:00
|
|
|
debug_break();
|
|
|
|
|
2014-09-30 04:12:27 +00:00
|
|
|
if (g_MemErrHandler)
|
2014-05-30 00:02:16 +00:00
|
|
|
{
|
|
|
|
#ifdef DEBUGGINGAIDS
|
2014-09-30 04:12:27 +00:00
|
|
|
g_MemErrHandler(g_MemErrLine, g_MemErrFile, g_MemErrFunc);
|
2014-05-30 00:02:16 +00:00
|
|
|
#else
|
2014-09-30 04:12:27 +00:00
|
|
|
g_MemErrHandler(0, "???", "???");
|
2014-05-30 00:02:16 +00:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2014-09-30 04:12:27 +00:00
|
|
|
Bexit(EXIT_FAILURE);
|
2018-10-25 23:30:22 +00:00
|
|
|
EDUKE32_UNREACHABLE_SECTION(return &handle_memerr);
|
2014-05-30 00:02:16 +00:00
|
|
|
}
|
|
|
|
|
2014-09-30 04:12:27 +00:00
|
|
|
void set_memerr_handler(void(*handlerfunc)(int32_t, const char *, const char *))
|
2014-05-30 00:02:16 +00:00
|
|
|
{
|
2014-09-30 04:12:27 +00:00
|
|
|
g_MemErrHandler = handlerfunc;
|
2014-05-30 00:02:16 +00:00
|
|
|
}
|
|
|
|
|
2006-04-23 06:44:19 +00:00
|
|
|
//
|
|
|
|
// Stuff which must be a function
|
|
|
|
//
|
|
|
|
char *Bgethomedir(void)
|
|
|
|
{
|
|
|
|
#ifdef _WIN32
|
2006-04-24 19:04:22 +00:00
|
|
|
|
2019-09-22 21:15:46 +00:00
|
|
|
char appdata[MAX_PATH];
|
2008-08-23 03:12:23 +00:00
|
|
|
|
2019-09-22 21:15:46 +00:00
|
|
|
if (SUCCEEDED(SHGetSpecialFolderPathA(NULL, appdata, CSIDL_APPDATA, FALSE)))
|
2018-10-25 23:30:17 +00:00
|
|
|
{
|
2019-09-22 21:15:46 +00:00
|
|
|
return Xstrdup(appdata);
|
2018-10-25 23:30:17 +00:00
|
|
|
}
|
2006-04-24 19:04:22 +00:00
|
|
|
return NULL;
|
2015-02-14 07:26:10 +00:00
|
|
|
#elif defined EDUKE32_OSX
|
2014-12-08 04:31:57 +00:00
|
|
|
return osx_gethomedir();
|
2006-04-23 06:44:19 +00:00
|
|
|
#else
|
2006-04-24 19:04:22 +00:00
|
|
|
char *e = getenv("HOME");
|
|
|
|
if (!e) return NULL;
|
2018-10-25 23:28:56 +00:00
|
|
|
return Xstrdup(e);
|
2006-04-23 06:44:19 +00:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2006-07-01 01:40:18 +00:00
|
|
|
|
2009-01-09 09:29:17 +00:00
|
|
|
int32_t Bcorrectfilename(char *filename, int32_t removefn)
|
2006-04-23 06:44:19 +00:00
|
|
|
{
|
2018-10-25 23:30:17 +00:00
|
|
|
char *fn = Xstrdup(filename);
|
|
|
|
char *tokarr[64], *first, *next = NULL;
|
2006-04-24 19:04:22 +00:00
|
|
|
|
2007-12-12 17:42:14 +00:00
|
|
|
for (first=fn; *first; first++)
|
|
|
|
{
|
2006-04-23 06:44:19 +00:00
|
|
|
#ifdef _WIN32
|
2006-04-24 19:04:22 +00:00
|
|
|
if (*first == '\\') *first = '/';
|
2006-04-23 06:44:19 +00:00
|
|
|
#endif
|
2006-04-24 19:04:22 +00:00
|
|
|
}
|
2018-10-25 23:30:17 +00:00
|
|
|
|
|
|
|
int leadslash = (*fn == '/');
|
|
|
|
int trailslash = (first>fn && first[-1] == '/');
|
|
|
|
int ntok = 0;
|
2006-04-24 19:04:22 +00:00
|
|
|
|
|
|
|
first = fn;
|
2007-12-12 17:42:14 +00:00
|
|
|
do
|
|
|
|
{
|
2018-10-25 23:30:17 +00:00
|
|
|
char *token = Bstrtoken(first, "/", &next, 1);
|
2006-04-24 19:04:22 +00:00
|
|
|
first = NULL;
|
|
|
|
if (!token) break;
|
|
|
|
else if (token[0] == 0) continue;
|
|
|
|
else if (token[0] == '.' && token[1] == 0) continue;
|
|
|
|
else if (token[0] == '.' && token[1] == '.' && token[2] == 0) ntok = max(0,ntok-1);
|
|
|
|
else tokarr[ntok++] = token;
|
2007-12-12 17:42:14 +00:00
|
|
|
}
|
|
|
|
while (1);
|
2006-04-24 19:04:22 +00:00
|
|
|
|
2007-12-12 17:42:14 +00:00
|
|
|
if (!trailslash && removefn) { ntok = max(0,ntok-1); trailslash = 1; }
|
2006-04-24 19:04:22 +00:00
|
|
|
if (ntok == 0 && trailslash && leadslash) trailslash = 0;
|
|
|
|
|
|
|
|
first = filename;
|
|
|
|
if (leadslash) *(first++) = '/';
|
2018-10-25 23:30:17 +00:00
|
|
|
for (int i=0; i<ntok; i++)
|
2007-12-12 17:42:14 +00:00
|
|
|
{
|
2006-04-24 19:04:22 +00:00
|
|
|
if (i>0) *(first++) = '/';
|
2018-10-25 23:30:17 +00:00
|
|
|
for (char *token=tokarr[i]; *token; token++)
|
2006-04-24 19:04:22 +00:00
|
|
|
*(first++) = *token;
|
|
|
|
}
|
|
|
|
if (trailslash) *(first++) = '/';
|
|
|
|
*(first++) = 0;
|
2006-04-23 06:44:19 +00:00
|
|
|
|
2019-06-25 11:29:08 +00:00
|
|
|
Xfree(fn);
|
2006-04-24 19:04:22 +00:00
|
|
|
return 0;
|
2006-04-23 06:44:19 +00:00
|
|
|
}
|
|
|
|
|
2015-01-11 04:53:30 +00:00
|
|
|
|
|
|
|
char *Bstrtoken(char *s, const char *delim, char **ptrptr, int chop)
|
2006-04-23 06:44:19 +00:00
|
|
|
{
|
2015-01-11 04:53:30 +00:00
|
|
|
if (!ptrptr)
|
|
|
|
return NULL;
|
2006-04-24 19:04:22 +00:00
|
|
|
|
2015-01-11 04:53:30 +00:00
|
|
|
char *p = s ? s : *ptrptr;
|
2006-04-23 06:44:19 +00:00
|
|
|
|
2015-01-11 04:53:30 +00:00
|
|
|
if (!p)
|
|
|
|
return NULL;
|
2006-04-23 06:44:19 +00:00
|
|
|
|
2015-01-11 04:53:30 +00:00
|
|
|
while (*p != 0 && Bstrchr(delim, *p)) p++;
|
2006-04-23 06:44:19 +00:00
|
|
|
|
2007-12-12 17:42:14 +00:00
|
|
|
if (*p == 0)
|
|
|
|
{
|
2006-04-24 19:04:22 +00:00
|
|
|
*ptrptr = NULL;
|
|
|
|
return NULL;
|
|
|
|
}
|
2015-01-11 04:53:30 +00:00
|
|
|
|
|
|
|
char * const start = p;
|
|
|
|
|
|
|
|
while (*p != 0 && !Bstrchr(delim, *p)) p++;
|
|
|
|
|
|
|
|
if (*p == 0)
|
|
|
|
*ptrptr = NULL;
|
2007-12-12 17:42:14 +00:00
|
|
|
else
|
|
|
|
{
|
2015-01-11 04:53:30 +00:00
|
|
|
if (chop)
|
|
|
|
*(p++) = 0;
|
2006-04-24 19:04:22 +00:00
|
|
|
*ptrptr = p;
|
|
|
|
}
|
2006-04-23 06:44:19 +00:00
|
|
|
|
2006-04-24 19:04:22 +00:00
|
|
|
return start;
|
2006-04-23 06:44:19 +00:00
|
|
|
}
|
|
|
|
|
2012-03-12 04:48:42 +00:00
|
|
|
char *Bstrtolower(char *str)
|
|
|
|
{
|
2015-01-11 04:53:30 +00:00
|
|
|
if (!str)
|
|
|
|
return NULL;
|
2012-03-12 04:48:42 +00:00
|
|
|
|
2015-01-11 04:53:30 +00:00
|
|
|
int len = Bstrlen(str);
|
2012-03-12 04:48:42 +00:00
|
|
|
|
2015-01-11 04:53:30 +00:00
|
|
|
if (len <= 0)
|
|
|
|
return str;
|
2012-03-12 04:48:42 +00:00
|
|
|
|
2015-01-11 04:53:30 +00:00
|
|
|
int i = 0;
|
|
|
|
|
|
|
|
do
|
|
|
|
{
|
|
|
|
*(str + i) = Btolower(*(str + i));
|
|
|
|
i++;
|
|
|
|
} while (--len);
|
2012-03-12 04:48:42 +00:00
|
|
|
|
|
|
|
return str;
|
|
|
|
}
|
|
|
|
|
2006-04-23 06:44:19 +00:00
|
|
|
#if !defined(_WIN32)
|
|
|
|
char *Bstrlwr(char *s)
|
|
|
|
{
|
2006-04-24 19:04:22 +00:00
|
|
|
if (!s) return s;
|
2015-01-11 04:53:30 +00:00
|
|
|
char *t = s;
|
2007-12-12 17:42:14 +00:00
|
|
|
while (*t) { *t = Btolower(*t); t++; }
|
2006-04-24 19:04:22 +00:00
|
|
|
return s;
|
2006-04-23 06:44:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
char *Bstrupr(char *s)
|
|
|
|
{
|
2006-04-24 19:04:22 +00:00
|
|
|
if (!s) return s;
|
2015-01-11 04:53:30 +00:00
|
|
|
char *t = s;
|
2007-12-12 17:42:14 +00:00
|
|
|
while (*t) { *t = Btoupper(*t); t++; }
|
2006-04-24 19:04:22 +00:00
|
|
|
return s;
|
2006-04-23 06:44:19 +00:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|