/* sys_linux.c Linux-style Unix system driver. Copyright (C) 1996-1997 Id Software, Inc. Copyright (C) 1999,2000 Nelson Rush. Copyright (C) 1999,2000 contributors of the QuakeForge project Please see the file "AUTHORS" for a list of contributors 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: Free Software Foundation, Inc. 59 Temple Place - Suite 330 Boston, MA 02111-1307, USA $Id$ */ #include #include #include #include #include #include #include #include #include #ifdef HAVE_UNISTD_H #include #endif #include #include #include #include #include #include #include #include #include #include #ifdef UQUAKE qboolean isDedicated; #endif extern int nostdout; int noconinput = 0; char *basedir = "."; char *cachedir = "/tmp"; // ======================================================================= // General routines // ======================================================================= void Sys_DebugNumber ( int y, int val ) { } void Sys_Quit ( void ) { Host_Shutdown(); exit(0); } void Sys_Init ( void ) { #if id386 Sys_SetFPCW(); #endif } void Sys_Error ( char *error, ... ) { va_list argptr; char string[1024]; va_start (argptr, error); vsnprintf (string, sizeof(string), error, argptr); va_end (argptr); fprintf(stderr, "Error: %s\n", string); Host_Shutdown (); exit (1); } void Sys_Warn ( char *warning, ... ) { va_list argptr; char string[1024]; va_start (argptr, warning); vsnprintf (string, sizeof(string), warning, argptr); va_end (argptr); fprintf(stderr, "Warning: %s", string); } int Sys_FileOpenRead ( char *path, int *handle ) { int h; struct stat fileinfo; h = open (path, O_RDONLY, 0666); *handle = h; if (h == -1) return -1; if (fstat (h,&fileinfo) == -1) Sys_Error ("Error fstating %s", path); return fileinfo.st_size; } int Sys_FileOpenWrite ( char *path ) { int handle; umask (0); handle = open(path,O_RDWR | O_CREAT | O_TRUNC, 0666); if (handle == -1) Sys_Error ("Error opening %s: %s", path,strerror(errno)); return handle; } int Sys_FileWrite ( int handle, void *src, int count ) { return write (handle, src, count); } void Sys_FileClose ( int handle ) { close (handle); } void Sys_FileSeek ( int handle, int position ) { lseek (handle, position, SEEK_SET); } int Sys_FileRead ( int handle, void *dest, int count ) { return read (handle, dest, count); } void Sys_DebugLog ( char *file, char *fmt, ... ) { va_list argptr; static char data[1024]; QFile *stream; unsigned char *p; // int fd; va_start(argptr, fmt); vsnprintf(data, sizeof(data), fmt, argptr); va_end(argptr); // fd = open(file, O_WRONLY | O_BINARY | O_CREAT | O_APPEND, 0666); stream = Qopen(file, "a"); for (p = (unsigned char *) data; *p; p++) { Qputc(stream, trans_table[*p]); } Qclose(stream); /* fd = open(file, O_WRONLY | O_CREAT | O_APPEND, 0666); write(fd, data, strlen(data)); close(fd); */ } /* This function isn't used. void Sys_EditFile ( char *filename ) { char cmd[256]; char *term; char *editor; term = getenv("TERM"); if (term && !strcmp(term, "xterm")) { editor = getenv("VISUAL"); if (!editor) editor = getenv("EDITOR"); if (!editor) editor = getenv("EDIT"); if (!editor) editor = "vi"; snprintf(cmd, sizeof(cmd), "xterm -e %s %s", editor, filename); system(cmd); } } */ // ======================================================================= // Sleeps for microseconds // ======================================================================= static volatile int oktogo; void alarm_handler ( int x ) { oktogo=1; } void floating_point_exception_handler ( int whatever ) { // Sys_Warn("floating point exception\n"); signal(SIGFPE, floating_point_exception_handler); } char * Sys_ConsoleInput ( void ) { #ifdef UQUAKE static char text[256]; fd_set fdset; int len; struct timeval timeout; if (cls.state == ca_dedicated) { FD_ZERO(&fdset); FD_SET(0, &fdset); // stdin timeout.tv_sec = 0; timeout.tv_usec = 0; if (select (1, &fdset, NULL, NULL, &timeout) == -1 || !FD_ISSET(0, &fdset)) return NULL; len = read (0, text, sizeof(text)); if (len < 1) return NULL; text[len-1] = 0; // rip off the \n and terminate return text; } #endif return NULL; } #if !id386 void Sys_HighFPPrecision ( void ) { } void Sys_LowFPPrecision ( void ) { } #endif #ifdef QUAKEWORLD int skipframes; #endif int main ( int c, char **v ) { double time, oldtime, newtime; quakeparms_t parms; int j; #ifdef UQUAKE extern int vcrFile; extern int recording; #endif signal(SIGFPE, SIG_IGN); memset(&parms, 0, sizeof(parms)); COM_InitArgv(c, v); parms.argc = com_argc; parms.argv = com_argv; parms.memsize = 16*1024*1024; j = COM_CheckParm("-mem"); if (j) parms.memsize = (int) (Q_atof(com_argv[j+1]) * 1024 * 1024); parms.membase = malloc (parms.memsize); if (!parms.membase) { printf ("Can't allocate zone memory!\n"); return 1; } parms.basedir = basedir; // caching is disabled by default, use -cachedir to enable // parms.cachedir = cachedir; noconinput = COM_CheckParm("-noconinput"); if (COM_CheckParm("-nostdout")) { nostdout = 1; } else { #ifdef QUAKEWORLD printf ("QuakeForge (QW Client) v%s\n", QF_VERSION); #else printf ("QuakeForge (UQuake) v%s\n", QF_VERSION); #endif } Sys_Init(); Host_Init(&parms); oldtime = Sys_DoubleTime (); while (1) { // find time spent rendering last frame newtime = Sys_DoubleTime (); time = newtime - oldtime; #ifdef UQUAKE if (cls.state == ca_dedicated) { // play vcrfiles at max speed if (time < sys_ticrate->value && (vcrFile == -1 || recording) ) { usleep(1); continue; // not time to run a server only tic yet } time = sys_ticrate->value; } if (time > sys_ticrate->value*2) oldtime = newtime; else oldtime += time; #else oldtime = newtime; #endif Host_Frame(time); } } /* Sys_MakeCodeWriteable */ void Sys_MakeCodeWriteable ( unsigned long startaddr, unsigned long length ) { int r; unsigned long addr; int psize = getpagesize(); addr = (startaddr & ~(psize-1)) - psize; // fprintf(stderr, "writable code %lx(%lx)-%lx, length=%lx\n", startaddr, // addr, startaddr+length, length); r = mprotect((char*)addr, length + startaddr - addr + psize, 7); if (r < 0) Sys_Error("Protection change failed\n"); }