This repository has been archived on 2023-07-11. You can view files and clone it, but cannot push or open issues or pull requests.
chocolate-doom-wii/pkg/osx/Execute.m
2016-10-30 18:40:00 -04:00

223 lines
5.4 KiB
Objective-C

//
// Copyright(C) 2005-2014 Simon Howard
//
// 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.
//
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <sys/wait.h>
#include <AppKit/AppKit.h>
#include "config.h"
#define RESPONSE_FILE "/tmp/launcher.rsp"
#define TEMP_SCRIPT "/tmp/tempscript.sh"
static char *executable_path;
// Called on startup to save the location of the launcher program
// (within a package, other executables should be in the same directory)
void SetProgramLocation(const char *path)
{
char *p;
executable_path = strdup(path);
p = strrchr(executable_path, '/');
*p = '\0';
}
// Write out the response file containing command line arguments.
static void WriteResponseFile(const char *iwad, const char *args)
{
FILE *fstream;
fstream = fopen(RESPONSE_FILE, "w");
if (iwad != NULL)
{
fprintf(fstream, "-iwad \"%s\"", iwad);
}
if (args != NULL)
{
fprintf(fstream, "%s", args);
}
fclose(fstream);
}
static void DoExec(const char *executable, const char *iwad, const char *args)
{
char *argv[3];
asprintf(&argv[0], "%s/%s", executable_path, executable);
if (iwad != NULL || args != NULL)
{
WriteResponseFile(iwad, args);
argv[1] = "@" RESPONSE_FILE;
argv[2] = NULL;
}
else
{
argv[1] = NULL;
}
execv(argv[0], argv);
exit(-1);
}
// Execute the specified executable contained in the same directory
// as the launcher, with the specified arguments.
void ExecuteProgram(const char *executable, const char *iwad, const char *args)
{
pid_t childpid;
char *homedir;
childpid = fork();
if (childpid == 0)
{
signal(SIGCHLD, SIG_DFL);
// Change directory to home dir before launch, so that any demos
// are saved somewhere sensible.
homedir = getenv("HOME");
if (homedir != NULL)
{
chdir(homedir);
}
DoExec(executable, iwad, args);
}
else
{
signal(SIGCHLD, SIG_IGN);
}
}
// Write a sequence of commands that will display the specified message
// via shell commands.
static void WriteMessage(FILE *script, char *msg)
{
char *p;
fprintf(script, "echo \"");
for (p=msg; *p != '\0'; ++p)
{
// Start new line?
if (*p == '\n')
{
fprintf(script, "\"\necho \"");
continue;
}
// Escaped character?
if (*p == '\\' || *p == '\"')
{
fprintf(script, "\\");
}
fprintf(script, "%c", *p);
}
fprintf(script, "\"\n");
}
// Open a terminal window with the PATH set appropriately, and DOOMWADPATH
// set to the specified value.
void OpenTerminalWindow(const char *doomwadpath)
{
FILE *stream;
// Generate a shell script that sets the PATH to include the location
// where the Doom binaries are, and DOOMWADPATH to include the
// IWAD files that have been configured in the launcher interface.
// The script then deletes itself and starts a shell.
stream = fopen(TEMP_SCRIPT, "w");
fprintf(stream, "#!/bin/sh\n");
//fprintf(stream, "set -x\n");
fprintf(stream, "PATH=\"%s:$PATH\"\n", executable_path);
// MANPATH is set to point to the directory within the bundle that
// contains the Unix manpages. However, the bundle name or path to
// it can contain a space, and OS X doesn't like this! As a
// workaround, create a symlink in /tmp to point to the real directory,
// and put *this* in MANPATH.
fprintf(stream, "rm -f \"/tmp/%s.man\"\n", PACKAGE_TARNAME);
fprintf(stream, "ln -s \"%s/man\" \"/tmp/%s.man\"\n",
executable_path, PACKAGE_TARNAME);
fprintf(stream, "MANPATH=\"/tmp/%s.man:$(manpath)\"\n", PACKAGE_TARNAME);
fprintf(stream, "export MANPATH\n");
fprintf(stream, "DOOMWADPATH=\"%s\"\n", doomwadpath);
fprintf(stream, "export DOOMWADPATH\n");
fprintf(stream, "rm -f \"%s\"\n", TEMP_SCRIPT);
// Display a useful message:
fprintf(stream, "clear\n");
WriteMessage(stream,
"\n"
"This command line has the PATH variable configured so that you may\n"
"launch the game with whatever parameters you desire.\n"
"\n"
"For example:\n"
"\n"
" " PACKAGE_TARNAME " -iwad doom2.wad -file sid.wad -warp 1\n"
"\n"
"Type 'exit' to exit.\n");
fprintf(stream, "exec $SHELL\n");
fprintf(stream, "\n");
fclose(stream);
chmod(TEMP_SCRIPT, 0755);
// Tell the terminal to open a window to run the script.
[[NSWorkspace sharedWorkspace] openFile: @TEMP_SCRIPT
withApplication: @"Terminal"];
}
void OpenDocumentation(const char *filename)
{
NSString *path;
path = [NSString stringWithFormat: @"%s/Documentation/%s",
executable_path, filename];
[[NSWorkspace sharedWorkspace] openFile: path];
}