spingle/source/unix/sys_sdl_unix.c

454 lines
8.5 KiB
C

/*
Copyright (C) 1996-2001 Id Software, Inc.
Copyright (C) 2002-2005 John Fitzgibbons and others
Copyright (C) 2007-2008 Kristian Duske
Copyright (C) 2010-2014 QuakeSpasm developers
This program is free software; you can redistribute it and/or
modify it under the terms of the GNU General Public License
as published by the Free Software Foundation; either version 2
of the License, or (at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
See the GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
*/
#include "q_defs.h"
#include <sys/types.h>
#include <errno.h>
#include <unistd.h>
#if PLATFORM_IS(OSX)
#include <libgen.h> /* dirname() and basename() */
#endif
#include <sys/stat.h>
#include <sys/time.h>
#include <fcntl.h>
#include <time.h>
#include <pwd.h>
#include <SDL.h>
bool isDedicated;
cvar_t sys_throttle = {"sys_throttle", "0.02", CVAR_ARCHIVE};
#define MAX_HANDLES 32 /* johnfitz -- was 10 */
static FILE *sys_handles[MAX_HANDLES];
static int32_t findhandle(void)
{
int32_t i;
for(i = 1; i < MAX_HANDLES; i++)
{
if(!sys_handles[i])
return i;
}
Sys_Error("out of handles");
return -1;
}
long Sys_filelength(FILE *f)
{
long pos, end;
pos = ftell(f);
fseek(f, 0, SEEK_END);
end = ftell(f);
fseek(f, pos, SEEK_SET);
return end;
}
int32_t Sys_FileOpenRead(const char *path, int32_t *hndl)
{
FILE *f;
int32_t i, retval;
i = findhandle();
f = fopen(path, "rb");
if(!f)
{
*hndl = -1;
retval = -1;
}
else
{
sys_handles[i] = f;
*hndl = i;
retval = Sys_filelength(f);
}
return retval;
}
int32_t Sys_FileOpenWrite(const char *path)
{
FILE *f;
int32_t i;
i = findhandle();
f = fopen(path, "wb");
if(!f)
Sys_Error("Error opening %s: %s", path, strerror(errno));
sys_handles[i] = f;
return i;
}
void Sys_FileClose(int32_t handle)
{
fclose(sys_handles[handle]);
sys_handles[handle] = NULL;
}
void Sys_FileSeek(int32_t handle, int32_t position)
{
fseek(sys_handles[handle], position, SEEK_SET);
}
int32_t Sys_FileRead(int32_t handle, void *dest, int32_t count)
{
return fread(dest, 1, count, sys_handles[handle]);
}
int32_t Sys_FileWrite(int32_t handle, const void *data, int32_t count)
{
return fwrite(data, 1, count, sys_handles[handle]);
}
int32_t Sys_FileTime(const char *path)
{
FILE *f;
f = fopen(path, "rb");
if(f)
{
fclose(f);
return 1;
}
return -1;
}
#if PLATFORM_IS(LINUX)
static int32_t Sys_NumCPUs(void)
{
int32_t numcpus = sysconf(_SC_NPROCESSORS_ONLN);
return (numcpus < 1) ? 1 : numcpus;
}
#elif PLATFORM_IS(OSX)
#include <sys/sysctl.h>
static int32_t Sys_NumCPUs(void)
{
int32_t numcpus;
int32_t mib[2];
size_t len;
numcpus = sysconf(_SC_NPROCESSORS_ONLN);
if(numcpus != -1)
return (numcpus < 1) ? 1 : numcpus;
len = sizeof(numcpus);
mib[0] = CTL_HW;
mib[1] = HW_AVAILCPU;
sysctl(mib, 2, &numcpus, &len, NULL, 0);
if(sysctl(mib, 2, &numcpus, &len, NULL, 0) == -1)
{
mib[1] = HW_NCPU;
if(sysctl(mib, 2, &numcpus, &len, NULL, 0) == -1)
return 1;
}
return (numcpus < 1) ? 1 : numcpus;
}
#elif PLATFORM_IS(BSD)
#include <sys/sysctl.h>
static int32_t Sys_NumCPUs(void)
{
int32_t numcpus;
int32_t mib[2];
size_t len;
numcpus = sysconf(_SC_NPROCESSORS_ONLN);
if(numcpus != -1)
return (numcpus < 1) ? 1 : numcpus;
len = sizeof(numcpus);
mib[0] = CTL_HW;
mib[1] = HW_NCPU;
if(sysctl(mib, 2, &numcpus, &len, NULL, 0) == -1)
return 1;
return (numcpus < 1) ? 1 : numcpus;
}
#else
static int32_t Sys_NumCPUs(void)
{
return -2;
}
#endif
static char const *Sys_GetHomeDir(void)
{
struct passwd *pwent;
pwent = getpwuid(getuid());
if(pwent == NULL)
{
perror("getpwuid");
return NULL;
}
return pwent->pw_dir ? pwent->pw_dir : getenv("HOME");
}
static char const *Sys_GetUserDir(void)
{
static char user_dir[MAX_OSPATH];
static char app_name[] = "spingle";
char const *home_dir;
#if PLATFORM_IS(OSX)
if(!(home_dir = Sys_GetHomeDir()))
return NULL;
q_snprintf(user_dir, sizeof(user_dir), "%s/Library/Application Support/%s", home_dir, app_name);
#else
char const *system_dir;
if((system_dir = getenv("XDG_DATA_HOME")))
q_snprintf(user_dir, sizeof(user_dir), "%s/%s/%s", home_dir, system_dir, app_name);
else
{
if(!(home_dir = Sys_GetHomeDir()))
return NULL;
q_snprintf(user_dir, sizeof(user_dir), "%s/.local/share/%s", home_dir, app_name);
}
#endif
return user_dir;
}
#if PLATFORM_IS(OSX)
static char *OSX_StripAppBundle(char *dir)
{
/* based on the ioq3 project at icculus.org. */
static char osx_path[MAX_OSPATH];
q_strlcpy(osx_path, dir, sizeof(osx_path));
if(strcmp(basename(osx_path), "MacOS"))
return dir;
q_strlcpy(osx_path, dirname(osx_path), sizeof(osx_path));
if(strcmp(basename(osx_path), "Contents"))
return dir;
q_strlcpy(osx_path, dirname(osx_path), sizeof(osx_path));
if(!strstr(basename(osx_path), ".app"))
return dir;
q_strlcpy(osx_path, dirname(osx_path), sizeof(osx_path));
return osx_path;
}
static void Sys_GetBasedir(char *argv0, char *dst, size_t dstsize)
{
char *tmp;
if(realpath(argv0, dst) == NULL)
{
perror("realpath");
if(getcwd(dst, dstsize - 1) == NULL)
_fail:
Sys_Error("Couldn't determine current directory");
}
else
{
/* strip off the binary name */
if(!(tmp = strdup(dst))) goto _fail;
q_strlcpy(dst, dirname(tmp), dstsize);
free(tmp);
}
tmp = OSX_StripAppBundle(dst);
if(tmp != dst)
q_strlcpy(dst, tmp, dstsize);
}
#else
static void Sys_GetBasedir(char *argv0, char *dst, size_t dstsize)
{
char *tmp;
(void)argv0;
if(getcwd(dst, dstsize - 1) == NULL)
Sys_Error("Couldn't determine current directory");
tmp = dst;
while(*tmp != 0)
tmp++;
while(*tmp == 0 && tmp != dst)
{
--tmp;
if(tmp != dst && *tmp == '/')
*tmp = 0;
}
}
#endif
void Sys_Init(void)
{
static char cwd[MAX_OSPATH];
char const *user_dir;
/* set base dir to current directory */
memset(cwd, 0, sizeof(cwd));
Sys_GetBasedir(host_parms->argv[0], cwd, sizeof(cwd));
host_parms->basedir = cwd;
/* check for user directory, or use base dir */
if((user_dir = Sys_GetUserDir()))
{
Sys_mkdir(user_dir);
host_parms->userdir = user_dir;
}
else
host_parms->userdir = host_parms->basedir;
host_parms->numcpus = Sys_NumCPUs();
Sys_Printf("Detected %" PRIi32 " CPUs\n", host_parms->numcpus);
}
void Sys_mkdir(const char *path)
{
int32_t rc = mkdir(path, 0777);
if(rc != 0 && errno == EEXIST)
{
struct stat st;
if(stat(path, &st) == 0 && S_ISDIR(st.st_mode))
rc = 0;
}
if(rc != 0)
{
rc = errno;
Sys_Error("Unable to create directory %s: %s", path, strerror(rc));
}
}
static const char errortxt1[] = "\nERROR-OUT BEGIN\n\n";
static const char errortxt2[] = "\nQUAKE ERROR: ";
void Sys_Error(const char *error, ...)
{
va_list argptr;
char text[1024];
host_parms->errstate++;
va_start(argptr, error);
q_vsnprintf(text, sizeof(text), error, argptr);
va_end(argptr);
fputs(errortxt1, stderr);
Host_Shutdown();
fputs(errortxt2, stderr);
fputs(text, stderr);
fputs("\n\n", stderr);
if(!isDedicated)
PL_ErrorDialog(text);
exit(1);
}
void Sys_Printf(const char *fmt, ...)
{
va_list argptr;
va_start(argptr, fmt);
vprintf(fmt, argptr);
va_end(argptr);
}
void Sys_Quit(void)
{
Host_Shutdown();
exit(0);
}
double Sys_DoubleTime(void)
{
return SDL_GetTicks() / 1000.0;
}
const char *Sys_ConsoleInput(void)
{
static char con_text[256];
static int32_t textlen;
char c;
fd_set set;
struct timeval timeout;
FD_ZERO(&set);
FD_SET(0, &set); // stdin
timeout.tv_sec = 0;
timeout.tv_usec = 0;
while(select(1, &set, NULL, NULL, &timeout))
{
read(0, &c, 1);
if(c == '\n' || c == '\r')
{
con_text[textlen] = '\0';
textlen = 0;
return con_text;
}
else if(c == 8)
{
if(textlen)
{
textlen--;
con_text[textlen] = '\0';
}
continue;
}
con_text[textlen] = c;
textlen++;
if(textlen < (int32_t) sizeof(con_text))
con_text[textlen] = '\0';
else
{
// buffer is full
textlen = 0;
con_text[0] = '\0';
Sys_Printf("\nConsole input too long!\n");
break;
}
}
return NULL;
}
void Sys_Sleep(uint32_t msecs)
{
SDL_Delay(msecs);
}
void Sys_SendKeyEvents(void)
{
IN_Commands(); //ericw -- allow joysticks to add keys so they can be used to confirm SCR_ModalMessage
IN_SendKeyEvents();
}