/* Copyright (C) 1997-2001 Id Software, Inc. 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 #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "../qcommon/qcommon.h" #include "../solaris/rw_solaris.h" #if defined(SOL8_XIL_WORKAROUND) && !defined(DEDICATED_ONLY) #include typedef struct { qboolean restart_sound; int s_rate; int s_width; int s_channels; int width; int height; byte *pic; byte *pic_pending; // order 1 huffman stuff int *hnodes1; // [256][256][2]; int numhnodes1[256]; int h_used[512]; int h_count[512]; } cinematics_t; #endif cvar_t *nostdout; unsigned sys_frame_time; uid_t saved_euid; qboolean stdin_active = true; hrtime_t base_hrtime; char display_name[ 1024 ]; // ======================================================================= // General routines // ======================================================================= void Sys_ConsoleOutput (char *string) { if (nostdout && nostdout->value) return; fputs(string, stdout); } void Sys_Printf (char *fmt, ...) { va_list argptr; char text[1024]; unsigned char *p; va_start (argptr,fmt); vsnprintf (text,1024,fmt,argptr); va_end (argptr); if (strlen(text) > sizeof(text)) Sys_Error("memory overwrite in Sys_Printf"); if (nostdout && nostdout->value) return; for (p = (unsigned char *)text; *p; p++) { *p &= 0x7f; if ((*p > 128 || *p < 32) && *p != 10 && *p != 13 && *p != 9) printf("[%02x]", *p); else putc(*p, stdout); } } void Sys_Quit (void) { CL_Shutdown (); Qcommon_Shutdown (); fcntl (0, F_SETFL, fcntl (0, F_GETFL, 0) & ~FNDELAY); _exit(0); } void Sys_Init(void) { #if id386 // Sys_SetFPCW(); #endif } void Sys_Error (char *error, ...) { /* DEBUG: matt */ /* exit(0); */ /* DEBUG */ va_list argptr; char string[1024]; // change stdin to non blocking fcntl (0, F_SETFL, fcntl (0, F_GETFL, 0) & ~FNDELAY); CL_Shutdown (); Qcommon_Shutdown (); va_start (argptr,error); vsnprintf (string,1024,error,argptr); va_end (argptr); fprintf(stderr, "Error: %s\n", string); _exit (1); } void Sys_Warn (char *warning, ...) { va_list argptr; char string[1024]; va_start (argptr,warning); vsnprintf (string,1024,warning,argptr); va_end (argptr); fprintf(stderr, "Warning: %s", string); } /* ============ Sys_FileTime returns -1 if not present ============ */ int Sys_FileTime (char *path) { struct stat buf; if (stat (path,&buf) == -1) return -1; return buf.st_mtime; } void floating_point_exception_handler(int whatever) { // Sys_Warn("floating point exception\n"); signal(SIGFPE, floating_point_exception_handler); } char *Sys_ConsoleInput(void) { static char text[256]; int len; fd_set fdset; struct timeval timeout; if (!dedicated || !dedicated->value) return NULL; if (!stdin_active) return NULL; 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 == 0) { // eof! stdin_active = false; return NULL; } if (len < 1) return NULL; text[len-1] = 0; // rip off the /n and terminate return text; } /*****************************************************************************/ static void *game_library; /* ================= Sys_UnloadGame ================= */ void Sys_UnloadGame (void) { if (game_library) dlclose (game_library); game_library = NULL; } /* ================= Sys_GetGameAPI Loads the game dll ================= */ void *Sys_GetGameAPI (void *parms) { void *(*GetGameAPI) (void *); FILE *fp; char name[MAX_OSPATH]; char curpath[MAX_OSPATH]; char *path; char *str_p; #ifdef __i386__ const char *gamename = "gamei386.so"; #elif defined(__sun__) || defined(sun) const char *gamename = "gamesparc.so"; #else #error Unknown arch #endif if (game_library) Com_Error (ERR_FATAL, "Sys_GetGameAPI without Sys_UnloadingGame"); getcwd(curpath, sizeof(curpath)); Com_Printf("------- Loading %s -------\n", gamename); // now run through the search paths path = NULL; while (1) { path = FS_NextPath (path); if (!path) return NULL; // couldn't find one anywhere snprintf (name, MAX_OSPATH, "%s/%s", path, gamename); /* skip it if it just doesn't exist */ fp = fopen(name, "rb"); if (fp == NULL) continue; fclose(fp); game_library = dlopen (name, RTLD_NOW); if (game_library) { Com_MDPrintf ("LoadLibrary (%s)\n",name); break; } else { Com_Printf ("LoadLibrary (%s):", name); path = dlerror(); str_p = strchr(path, ':'); // skip the path (already shown) if (str_p == NULL) str_p = path; else str_p++; Com_Printf ("%s\n", str_p); return NULL; } } GetGameAPI = (void *)dlsym (game_library, "GetGameAPI"); if (!GetGameAPI) { Sys_UnloadGame (); return NULL; } return GetGameAPI (parms); } /*****************************************************************************/ void Sys_AppActivate (void) { } void Sys_SendKeyEvents (void) { #ifndef DEDICATED_ONLY if (KBD_Update_fp) KBD_Update_fp(); #endif // grab frame time sys_frame_time = Sys_Milliseconds(); } /*****************************************************************************/ char *Sys_GetClipboardData( void ) { return NULL; } #if defined(SOL8_XIL_WORKAROUND) && !defined(DEDICATED_ONLY) XilSystemState xil_state; #endif int main( int argc, char **argv ) { #ifdef DEDICATED_ONLY int newargc; char **newargv; int i; #endif int time, oldtime, newtime; sigset_t sigs; #if defined(SOL8_XIL_WORKAROUND) && !defined(DEDICATED_ONLY) { extern cinematics_t cin; if( (xil_state = xil_open()) == NULL ) { fprintf( stderr, "can't open XIL\n" ); exit( 1 ); } memset( &cin, 0, sizeof( cin ) ); } #endif /* * Save the contents of the DISPLAY environment variable. * if we don't, it gets overwritten after reloading the * renderer libraries. */ { char *tmp_name = getenv( "DISPLAY" ); if( tmp_name == NULL ) { display_name[ 0 ] = 0; } else { strncpy( display_name, tmp_name, sizeof( display_name ) ); } } /* block the SIGPOLL signal so that only the audio thread gets it */ sigemptyset( &sigs ); sigaddset( &sigs, SIGPOLL ); pthread_sigmask( SIG_BLOCK, &sigs, NULL ); // go back to real user for config loads saved_euid = geteuid(); seteuid( getuid() ); base_hrtime = gethrtime(); #ifdef DEDICATED_ONLY // force dedicated newargc = argc; newargv = malloc((argc + 3) * sizeof(char *)); newargv[0] = argv[0]; newargv[1] = "+set"; newargv[2] = "dedicated"; newargv[3] = "1"; for (i = 1; i < argc; i++) newargv[i + 3] = argv[i]; newargc += 3; Qcommon_Init(newargc, newargv); #else Qcommon_Init(argc, argv); #endif fcntl(0, F_SETFL, fcntl (0, F_GETFL, 0) | FNDELAY); nostdout = Cvar_Get("nostdout", "0", 0); if (!nostdout->value) { fcntl(0, F_SETFL, fcntl (0, F_GETFL, 0) | FNDELAY); // printf ("Solaris Quake 2 -- Version %0.3f\n", SOLARIS_VERSION); } oldtime = Sys_Milliseconds (); while (1) { // find time spent rendering last frame do { newtime = Sys_Milliseconds (); time = newtime - oldtime; } while (time < 1); Qcommon_Frame (time); oldtime = newtime; } } void Sys_CopyProtect(void) { return; } #if 0 /* ================ 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"); } #endif size_t verify_fread( void *ptr, size_t size, size_t nitems, FILE *fp ) { size_t ret; int err; clearerr( fp ); ret = fread( ptr, size, nitems, fp ); err = errno; if( ret != nitems ) { printf( "verify_fread(...,%d,%d,...): return value: %d\n", size, nitems, ret ); if( ret == 0 && ferror( fp ) ) { printf( " error: %s\n", strerror( err ) ); printf( " fileno=%d\n", fileno( fp ) ); } /* sleep( 5 );*/ } return ret; } size_t verify_fwrite( void *ptr, size_t size, size_t nitems, FILE *fp ) { size_t ret; int err; clearerr( fp ); ret = fwrite( ptr, size, nitems, fp ); err = errno; if( ret != nitems ) { printf( "verify_fwrite(...,%d,%d,...): return value: %d\n", size, nitems, ret ); if( ret == 0 && ferror( fp ) ) { printf( " error: %s\n", strerror( err ) ); printf( " fileno=%d\n", fileno( fp ) ); } /* sleep( 5 );*/ } return ret; }