/* SDL_main.c, placed in the public domain by Sam Lantinga 4/13/98 The WinMain function -- calls your program's main() function */ #include #include #define WIN32_LEAN_AND_MEAN #include #ifdef _WIN32_WCE # define DIR_SEPERATOR TEXT("\\") # undef _getcwd # define _getcwd(str,len) wcscpy(str,TEXT("")) # define setbuf(f,b) # define setvbuf(w,x,y,z) # define fopen _wfopen # define freopen _wfreopen # define remove(x) DeleteFile(x) #else # define DIR_SEPERATOR TEXT("/") # include #endif /* Include the SDL main definition header */ #include "SDL.h" #include "SDL_main.h" #ifdef main # ifndef _WIN32_WCE_EMULATION # undef main # endif /* _WIN32_WCE_EMULATION */ #endif /* main */ /* The standard output files */ #define STDOUT_FILE TEXT("stdout.txt") #define STDERR_FILE TEXT("stderr.txt") #ifndef NO_STDIO_REDIRECT # ifdef _WIN32_WCE static wchar_t stdoutPath[MAX_PATH]; static wchar_t stderrPath[MAX_PATH]; # else static char stdoutPath[MAX_PATH]; static char stderrPath[MAX_PATH]; # endif #endif #if defined(_WIN32_WCE) && _WIN32_WCE < 300 /* seems to be undefined in Win CE although in online help */ #define isspace(a) (((CHAR)a == ' ') || ((CHAR)a == '\t')) #endif /* _WIN32_WCE < 300 */ /* Parse a command line buffer into arguments */ static int ParseCommandLine (char *cmdline, char **argv) { char *bufp; int argc; argc = 0; for (bufp = cmdline; *bufp;) { /* Skip leading whitespace */ while (isspace (*bufp)) { ++bufp; } /* Skip over argument */ if (*bufp == '"') { ++bufp; if (*bufp) { if (argv) { argv[argc] = bufp; } ++argc; } /* Skip over word */ while (*bufp && (*bufp != '"')) { ++bufp; } } else { if (*bufp) { if (argv) { argv[argc] = bufp; } ++argc; } /* Skip over word */ while (*bufp && !isspace (*bufp)) { ++bufp; } } if (*bufp) { if (argv) { *bufp = '\0'; } ++bufp; } } if (argv) { argv[argc] = NULL; } return (argc); } /* Show an error message */ static void ShowError (const char *title, const char *message) { /* If USE_MESSAGEBOX is defined, you need to link with user32.lib */ #ifdef USE_MESSAGEBOX MessageBox (NULL, message, title, MB_ICONEXCLAMATION | MB_OK); #else fprintf (stderr, "%s: %s\n", title, message); #endif } /* Pop up an out of memory message, returns to Windows */ static BOOL OutOfMemory (void) { ShowError ("Fatal Error", "Out of memory - aborting"); return FALSE; } /* SDL_Quit() shouldn't be used with atexit() directly because calling conventions may differ... */ static void cleanup (void) { SDL_Quit (); } /* Remove the output files if there was no output written */ static void cleanup_output (void) { #ifndef NO_STDIO_REDIRECT FILE *file; int empty; #endif /* Flush the output in case anything is queued */ fclose (stdout); fclose (stderr); #ifndef NO_STDIO_REDIRECT /* See if the files have any output in them */ if (stdoutPath[0]) { file = fopen (stdoutPath, TEXT ("rb")); if (file) { empty = (fgetc (file) == EOF) ? 1 : 0; fclose (file); if (empty) { remove (stdoutPath); } } } if (stderrPath[0]) { file = fopen (stderrPath, TEXT ("rb")); if (file) { empty = (fgetc (file) == EOF) ? 1 : 0; fclose (file); if (empty) { remove (stderrPath); } } } #endif } #if defined(_MSC_VER) && !defined(_WIN32_WCE) /* The VC++ compiler needs main defined */ #define console_main main #endif /* This is where execution begins [console apps] */ int console_main (int argc, char *argv[]) { size_t n; char *bufp, *appname; int status; /* Get the class name from argv[0] */ appname = argv[0]; if ((bufp = SDL_strrchr (argv[0], '\\')) != NULL) { appname = bufp + 1; } else if ((bufp = SDL_strrchr (argv[0], '/')) != NULL) { appname = bufp + 1; } if ((bufp = SDL_strrchr (appname, '.')) == NULL) n = SDL_strlen (appname); else n = (bufp - appname); bufp = SDL_stack_alloc (char, n + 1); if (bufp == NULL) { return OutOfMemory (); } SDL_strlcpy (bufp, appname, n + 1); appname = bufp; /* Load SDL dynamic link library */ if (SDL_Init (SDL_INIT_NOPARACHUTE) < 0) { ShowError ("WinMain() error", SDL_GetError ()); return (FALSE); } atexit (cleanup_output); atexit (cleanup); /* Sam: We still need to pass in the application handle so that DirectInput will initialize properly when SDL_RegisterApp() is called later in the video initialization. */ SDL_SetModuleHandle (GetModuleHandle (NULL)); /* Run the application main() code */ status = SDL_main (argc, argv); /* Exit cleanly, calling atexit() functions */ exit (status); /* Hush little compiler, don't you cry... */ return 0; } /* This is where execution begins [windowed apps] */ #ifdef _WIN32_WCE int WINAPI WinMain (HINSTANCE hInst, HINSTANCE hPrev, LPWSTR szCmdLine, int sw) #else int WINAPI WinMain (HINSTANCE hInst, HINSTANCE hPrev, LPSTR szCmdLine, int sw) #endif { HINSTANCE handle; char **argv; int argc; char *cmdline; DWORD pathlen; #ifdef _WIN32_WCE wchar_t path[MAX_PATH]; #else char path[MAX_PATH]; #endif #ifdef _WIN32_WCE wchar_t *bufp; int nLen; #else char *bufp; size_t nLen; #endif #ifndef NO_STDIO_REDIRECT FILE *newfp; #endif /* Start up DDHELP.EXE before opening any files, so DDHELP doesn't keep them open. This is a hack.. hopefully it will be fixed someday. DDHELP.EXE starts up the first time DDRAW.DLL is loaded. */ handle = LoadLibrary (TEXT ("DDRAW.DLL")); if (handle != NULL) { FreeLibrary (handle); } #ifndef NO_STDIO_REDIRECT pathlen = GetModuleFileName (NULL, path, SDL_arraysize (path)); while (pathlen > 0 && path[pathlen] != '\\') { --pathlen; } path[pathlen] = '\0'; #ifdef _WIN32_WCE wcsncpy (stdoutPath, path, SDL_arraysize (stdoutPath)); wcsncat (stdoutPath, DIR_SEPERATOR STDOUT_FILE, SDL_arraysize (stdoutPath)); #else SDL_strlcpy (stdoutPath, path, SDL_arraysize (stdoutPath)); SDL_strlcat (stdoutPath, DIR_SEPERATOR STDOUT_FILE, SDL_arraysize (stdoutPath)); #endif /* Redirect standard input and standard output */ newfp = freopen (stdoutPath, TEXT ("w"), stdout); #ifndef _WIN32_WCE if (newfp == NULL) { /* This happens on NT */ #if !defined(stdout) stdout = fopen (stdoutPath, TEXT ("w")); #else newfp = fopen (stdoutPath, TEXT ("w")); if (newfp) { *stdout = *newfp; } #endif } #endif /* _WIN32_WCE */ #ifdef _WIN32_WCE wcsncpy (stderrPath, path, SDL_arraysize (stdoutPath)); wcsncat (stderrPath, DIR_SEPERATOR STDOUT_FILE, SDL_arraysize (stdoutPath)); #else SDL_strlcpy (stderrPath, path, SDL_arraysize (stderrPath)); SDL_strlcat (stderrPath, DIR_SEPERATOR STDERR_FILE, SDL_arraysize (stderrPath)); #endif newfp = freopen (stderrPath, TEXT ("w"), stderr); #ifndef _WIN32_WCE if (newfp == NULL) { /* This happens on NT */ #if !defined(stderr) stderr = fopen (stderrPath, TEXT ("w")); #else newfp = fopen (stderrPath, TEXT ("w")); if (newfp) { *stderr = *newfp; } #endif } #endif /* _WIN32_WCE */ setvbuf (stdout, NULL, _IOLBF, BUFSIZ); /* Line buffered */ setbuf (stderr, NULL); /* No buffering */ #endif /* !NO_STDIO_REDIRECT */ #ifdef _WIN32_WCE nLen = wcslen (szCmdLine) + 128 + 1; bufp = SDL_stack_alloc (wchar_t, nLen * 2); wcscpy (bufp, TEXT ("\"")); GetModuleFileName (NULL, bufp + 1, 128 - 3); wcscpy (bufp + wcslen (bufp), TEXT ("\" ")); wcsncpy (bufp + wcslen (bufp), szCmdLine, nLen - wcslen (bufp)); nLen = wcslen (bufp) + 1; cmdline = SDL_stack_alloc (char, nLen); if (cmdline == NULL) { return OutOfMemory (); } WideCharToMultiByte (CP_ACP, 0, bufp, -1, cmdline, nLen, NULL, NULL); #else /* Grab the command line */ bufp = GetCommandLine (); nLen = SDL_strlen (bufp) + 1; cmdline = SDL_stack_alloc (char, nLen); if (cmdline == NULL) { return OutOfMemory (); } SDL_strlcpy (cmdline, bufp, nLen); #endif /* Parse it into argv and argc */ argc = ParseCommandLine (cmdline, NULL); argv = SDL_stack_alloc (char *, argc + 1); if (argv == NULL) { return OutOfMemory (); } ParseCommandLine (cmdline, argv); /* Run the main program (after a little SDL initialization) */ console_main (argc, argv); /* Hush little compiler, don't you cry... */ return 0; } /* vi: set ts=4 sw=4 expandtab: */