/** \file osxhelp.c * use OSX Help system */ /* XTrkCad - Model Railroad CAD * Copyright (C) 2015 Martin Fischer * * 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA */ #include #include #include #include #include #include #include #include "gtkint.h" #include "i18n.h" #define HELPCOMMANDPIPE "/tmp/helppipe" #define EXITCOMMAND "##exit##" #define HELPERPROGRAM "helphelper" static pid_t pidOfChild; static int handleOfPipe; extern char *wExecutableName; extern wBool_t CheckHelpTopicExists(const char *); /** * Create the fully qualified filename for the help helper * * \param parent IN path and filename of parent process * \return filename for the child process */ static char *ChildProgramFile(char *parentProgram) { char *startOfFilename; char *childProgram; childProgram = malloc(strlen(parentProgram)+ sizeof(HELPERPROGRAM) + 1); strcpy(childProgram, parentProgram); startOfFilename = strrchr(childProgram, '/'); strcpy(startOfFilename + 1, HELPERPROGRAM); return (childProgram); } /** * Invoke the help system to display help for . * * \param topic IN topic string */ void wHelp(const char * topic) { pid_t newPid; int status; const char html[] = ".html"; static char *directory; /**< base directory for HTML files */ char * htmlFile; struct { int length; char *page; } buffer; if (!CheckHelpTopicExists(topic)) { return; } // check whether child already exists if (pidOfChild != 0) { if (waitpid(pidOfChild, &status, WNOHANG) < 0) { // child exited -> clean up close(handleOfPipe); unlink(HELPCOMMANDPIPE); handleOfPipe = 0; pidOfChild = 0; // child exited } } // (re)start child if (pidOfChild == 0) { unlink(HELPCOMMANDPIPE); int rc = mkfifo(HELPCOMMANDPIPE, 0666); newPid = fork(); /* New process starts here */ if (newPid > 0) { pidOfChild = newPid; } else if (newPid == 0) { char *child = ChildProgramFile(wExecutableName); if (execlp(child, child, NULL) < 0) { /* never normally returns */ exit(8); } free(child); } else { /* -1 signifies fork failure */ pidOfChild = 0; return; } } buffer.page = malloc(sizeof(int)+strlen(topic) + strlen(html) + 1); if (!buffer.page) { return; } strcpy(buffer.page, topic); strcat(buffer.page, html); buffer.length = strlen(buffer.page); if (buffer.length>255) { printf("Help Topic too long %s", buffer.page); return; } if (!handleOfPipe) { handleOfPipe = open(HELPCOMMANDPIPE, O_WRONLY); if (handleOfPipe < 0) { if (pidOfChild) { kill(pidOfChild, SIGKILL); /* tidy up on next call */ } handleOfPipe = 0; return; } } int written = 0; int towrite = sizeof(int); while (written < towrite) { written += write(handleOfPipe, &buffer.length, sizeof(int)); } written =0; towrite = strlen(buffer.page); while (written < towrite) { written += write(handleOfPipe, buffer.page+written, towrite-written); } fsync(handleOfPipe); free(buffer.page); }