/* * %CopyrightBegin% * * Copyright Ericsson AB 1998-2009. All Rights Reserved. * * The contents of this file are subject to the Erlang Public License, * Version 1.1, (the "License"); you may not use this file except in * compliance with the License. You should have received a copy of the * Erlang Public License along with this software. If not, it can be * retrieved online at http://www.erlang.org/. * * Software distributed under the License is distributed on an "AS IS" * basis, WITHOUT WARRANTY OF ANY KIND, either express or implied. See * the License for the specific language governing rights and limitations * under the License. * * %CopyrightEnd% */ /* * start_erl.c - Windows NT start_erl * * Author: Mattias Nilsson */ #define WIN32_LEAN_AND_MEAN #define STRICT #include #include #include #include #include char *progname; /* * If CASE_SENSITIVE_OPTIONS is specified, options are case sensitive * (lower case). * The reason for this switch is that _strnicmp is Microsoft specific. */ #ifndef CASE_SENSITIVE_OPTIONS #define strnicmp strncmp #else #define strnicmp _strnicmp #endif #define RELEASE_SUBDIR "\\releases" #define REGISTRY_BASE "Software\\Ericsson\\Erlang\\" #define DEFAULT_DATAFILE "start_erl.data" /* Global variables holding option values and command lines */ char *CommandLineStart = NULL; char *ErlCommandLine = NULL; char *MyCommandLine = NULL; char *DataFileName = NULL; char *RelDir = NULL; char *BootFlagsFile = NULL; char *BootFlags = NULL; char *RegistryKey = NULL; char *BinDir = NULL; char *RootDir = NULL; char *VsnDir = NULL; char *Version = NULL; char *Release = NULL; BOOL NoConfig=FALSE; PROCESS_INFORMATION ErlProcessInfo; /* * Error reason printout 'the microsoft way' */ void ShowLastError(void) { LPVOID lpMsgBuf; DWORD dwErr; dwErr = GetLastError(); if( dwErr == ERROR_SUCCESS ) return; FormatMessage( FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM, NULL, dwErr, MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), (LPTSTR) &lpMsgBuf, 0, NULL ); fprintf(stderr, lpMsgBuf); LocalFree( lpMsgBuf ); } /* * Exit the program and give a nice and firm explanation of why * and how you can avoid it. */ void exit_help(char *err) { ShowLastError(); fprintf(stderr, "** Error: %s\n", err); printf("Usage:\n%s\n" " [] ++\n" " [-data ]\n" " [-reldir ]\n" " [-bootflags ]\n" " [-noconfig]\n", progname); exit(0); } /* * Splits the command line into two strings: * 1. Options to the Erlang node (ErlCommandLine) * 2. Options to this program (MyCommandLine) */ void split_commandline(void) { char *cmdline = CommandLineStart; progname=cmdline; /* Remove the first (quoted) string (our program name) */ if(*cmdline == '"') { cmdline++; /* Skip " */ while( (*cmdline != '\0') && (*cmdline++) != '"' ) ; } else { while( (*cmdline != '\0') && (*cmdline++) != ' ' ) ; } while( (*cmdline) == ' ' ) cmdline++; if( *cmdline == '\0') { ErlCommandLine = ""; MyCommandLine = ""; return; } cmdline[-1] = '\0'; /* Start from the end of the string and search for "++ " (PLUS PLUS SPACE) */ ErlCommandLine = cmdline; if(strncmp(cmdline,"++ ",3)) cmdline = strstr(cmdline," ++ "); if( cmdline == NULL ) { MyCommandLine = ""; return; } /* Terminate the ErlCommandLine where MyCommandLine starts */ *cmdline++ = '\0'; /* Skip 'whitespace--whitespace' (WHITESPACE MINUS MINUS WHITESPACE) */ while( (*cmdline) == ' ' ) cmdline++; while( (*cmdline) == '+' ) cmdline++; while( (*cmdline) == ' ' ) cmdline++; MyCommandLine = cmdline; #ifdef _DEBUG fprintf(stderr, "ErlCommandLine: '%s'\n", ErlCommandLine); fprintf(stderr, "MyCommandLine: '%s'\n", MyCommandLine); #endif } /* * 'Smart' unquoting of a string: \" becomes " and " becomes (nothing) * Skips any leading spaces and parses up to NULL or end of quoted string. * Calls exit_help() if an unterminated quote is detected. */ char * unquote_optionarg(char *str, char **strp) { char *newstr = (char *)malloc(strlen(str)+1); /* This one is realloc:ed later */ int i=0, inquote=0; assert(newstr); assert(str); /* Skip leading spaces */ while( *str == ' ' ) str++; /* Loop while in quote or until EOS or unquoted space */ while( (inquote==1) || ( (*str!=0) && (*str!=' ') ) ) { switch( *str ) { case '\\': /* If we are inside a quoted string we should convert \c to c */ if( inquote && str[1] == '"' ) str++; newstr[i++]=*str++; break; case '"': inquote = 1-inquote; *str++; break; default: newstr[i++]=*str++; break; } if( (*str == 0) && (inquote==1) ) { exit_help("Unterminated quote."); } } newstr[i++] = 0x00; /* Update the supplied pointer (used for continued parsing of options) */ *strp = str; /* Adjust memblock of newstr */ newstr = (char *)realloc(newstr, i); assert(newstr); return(newstr); } /* * Parses MyCommandLine and tries to fill in all the required option variables * (one way or another). */ void parse_commandline(void) { char *cmdline = MyCommandLine; while( *cmdline != '\0' ) { switch( *cmdline ) { case '-': /* Handle both -arg and /arg */ case '/': *cmdline++; if( strnicmp(cmdline, "data", 4) == 0) { DataFileName = unquote_optionarg(cmdline+4, &cmdline); } else if( strnicmp(cmdline, "reldir", 6) == 0) { RelDir = unquote_optionarg(cmdline+6, &cmdline); #ifdef _DEBUG fprintf(stderr, "RelDir: '%s'\n", RelDir); #endif } else if( strnicmp(cmdline, "bootflags", 9) == 0) { BootFlagsFile = unquote_optionarg(cmdline+9, &cmdline); } else if( strnicmp(cmdline, "noconfig", 8) == 0) { NoConfig=TRUE; #ifdef _DEBUG fprintf(stderr, "NoConfig=TRUE\n"); #endif } else { fprintf(stderr, "Unkown option: '%s'\n", cmdline); exit_help("Unknown command line option"); } break; default: cmdline++; break; } } } /* * Read the data file specified and get the version and release number * from it. * * This function also construct the correct RegistryKey from the version information * retrieved. */ void read_datafile(void) { FILE *fp; char *newname; long size; if(!DataFileName){ DataFileName = malloc(strlen(DEFAULT_DATAFILE) + 1); strcpy(DataFileName,DEFAULT_DATAFILE); } /* Is DataFileName relative or absolute ? */ if( (DataFileName[0] != '\\') && (strncmp(DataFileName+1, ":\\", 2)!=0) ) { /* Relative name, we have to prepend RelDir to it. */ if( !RelDir ) { exit_help("Need -reldir when -data filename has relative path."); } else { newname = (char *)malloc(strlen(DataFileName)+strlen(RelDir)+2); assert(newname); sprintf(newname, "%s\\%s", RelDir, DataFileName); free(DataFileName); DataFileName=newname; } } #ifdef _DEBUG fprintf(stderr, "DataFileName: '%s'\n", DataFileName); #endif if( (fp=fopen(DataFileName, "rb")) == NULL) { exit_help("Cannot find the datafile."); } fseek(fp, 0, SEEK_END); size=ftell(fp); fseek(fp, 0, SEEK_SET); Version = (char *)malloc(size+1); Release = (char *)malloc(size+1); assert(Version); assert(Release); if( (fscanf(fp, "%s %s", Version, Release)) == 0) { fclose(fp); exit_help("Format error in datafile."); } fclose(fp); #ifdef _DEBUG fprintf(stderr, "DataFile version: '%s'\n", Version); fprintf(stderr, "DataFile release: '%s'\n", Release); #endif } /* * Read the registry keys we need */ void read_registry_keys(void) { HKEY hReg; ULONG lLen; /* Create the RegistryKey name */ RegistryKey = (char *) malloc(strlen(REGISTRY_BASE) + strlen(Version) + 1); assert(RegistryKey); sprintf(RegistryKey, REGISTRY_BASE "%s", Version); /* We always need to find BinDir */ if( (RegOpenKeyEx(HKEY_LOCAL_MACHINE, RegistryKey, 0, KEY_READ, &hReg)) != ERROR_SUCCESS ) { exit_help("Could not open registry key."); } /* First query size of data */ if( (RegQueryValueEx(hReg, "Bindir", NULL, NULL, NULL, &lLen)) != ERROR_SUCCESS) { exit_help("Failed to query BinDir of release.\n"); } /* Allocate enough space */ BinDir = (char *)malloc(lLen+1); assert(BinDir); /* Retrieve the value */ if( (RegQueryValueEx(hReg, "Bindir", NULL, NULL, (unsigned char *) BinDir, &lLen)) != ERROR_SUCCESS) { exit_help("Failed to query BinDir of release (2).\n"); } #ifdef _DEBUG fprintf(stderr, "Bindir: '%s'\n", BinDir); #endif /* We also need the rootdir, in case we need to build RelDir later */ /* First query size of data */ if( (RegQueryValueEx(hReg, "Rootdir", NULL, NULL, NULL, &lLen)) != ERROR_SUCCESS) { exit_help("Failed to query RootDir of release.\n"); } /* Allocate enough space */ RootDir = (char *) malloc(lLen+1); assert(RootDir); /* Retrieve the value */ if( (RegQueryValueEx(hReg, "Rootdir", NULL, NULL, (unsigned char *) RootDir, &lLen)) != ERROR_SUCCESS) { exit_help("Failed to query RootDir of release (2).\n"); } #ifdef _DEBUG fprintf(stderr, "Rootdir: '%s'\n", RootDir); #endif RegCloseKey(hReg); } /* * Read the bootflags. This file contains extra command line options to erl.exe */ void read_bootflags(void) { FILE *fp; long fsize; char *newname; if(BootFlagsFile) { /* Is BootFlagsFile relative or absolute ? */ if( (BootFlagsFile[0] != '\\') && (strncmp(BootFlagsFile+1, ":\\", 2)!=0) ) { /* Relative name, we have to prepend RelDir\\Version to it. */ if( !RelDir ) { exit_help("Need -reldir when -bootflags " "filename has relative path."); } else { newname = (char *)malloc(strlen(BootFlagsFile)+strlen(RelDir)+strlen(Release)+3); assert(newname); sprintf(newname, "%s\\%s\\%s", RelDir, Release, BootFlagsFile); free(BootFlagsFile); BootFlagsFile=newname; } } #ifdef _DEBUG fprintf(stderr, "BootFlagsFile: '%s'\n", BootFlagsFile); #endif if( (fp=fopen(BootFlagsFile, "rb")) == NULL) { exit_help("Could not open BootFlags file."); } fseek(fp, 0, SEEK_END); fsize=ftell(fp); fseek(fp, 0, SEEK_SET); BootFlags = (char *)malloc(fsize+1); assert(BootFlags); if( (fgets(BootFlags, fsize+1, fp)) == NULL) { exit_help("Error while reading BootFlags file"); } fclose(fp); /* Adjust buffer size */ BootFlags = (char *)realloc(BootFlags, strlen(BootFlags)+1); assert(BootFlags); /* Strip \r\n from BootFlags */ fsize = strlen(BootFlags); while( fsize > 0 && ( (BootFlags[fsize-1] == '\r') || (BootFlags[fsize-1] == '\n') ) ) { BootFlags[--fsize]=0; } } else { /* Set empty BootFlags */ BootFlags = ""; } #ifdef _DEBUG fprintf(stderr, "BootFlags: '%s'\n", BootFlags); #endif } long start_new_node(void) { char *CommandLine; unsigned long i; STARTUPINFO si; DWORD dwExitCode; i = strlen(RelDir) + strlen(Release) + 4; VsnDir = (char *)malloc(i); assert(VsnDir); sprintf(VsnDir, "%s\\%s", RelDir, Release); if( NoConfig ) { i = strlen(BinDir) + strlen(ErlCommandLine) + strlen(BootFlags) + 64; CommandLine = (char *)malloc(i); assert(CommandLine); sprintf(CommandLine, "\"%s\\erl.exe\" -boot \"%s\\start\" %s %s", BinDir, VsnDir, ErlCommandLine, BootFlags); } else { i = strlen(BinDir) + strlen(ErlCommandLine) + strlen(BootFlags) + strlen(VsnDir)*2 + 64; CommandLine = (char *)malloc(i); assert(CommandLine); sprintf(CommandLine, "\"%s\\erl.exe\" -boot \"%s\\start\" -config \"%s\\sys\" %s %s", BinDir, VsnDir, VsnDir, ErlCommandLine, BootFlags); } #ifdef _DEBUG fprintf(stderr, "CommandLine: '%s'\n", CommandLine); #endif /* Initialize the STARTUPINFO structure. */ memset(&si, 0, sizeof(STARTUPINFO)); si.cb = sizeof(STARTUPINFO); si.lpTitle = NULL; si.dwFlags = STARTF_USESTDHANDLES; si.hStdInput = GetStdHandle(STD_INPUT_HANDLE); si.hStdOutput = GetStdHandle(STD_OUTPUT_HANDLE); si.hStdError = GetStdHandle(STD_ERROR_HANDLE); /* Create the new Erlang process */ if( (CreateProcess( NULL, /* pointer to name of executable module */ CommandLine, /* pointer to command line string */ NULL, /* pointer to process security attributes */ NULL, /* pointer to thread security attributes */ TRUE, /* handle inheritance flag */ GetPriorityClass(GetCurrentProcess()), /* creation flags */ NULL, /* pointer to new environment block */ BinDir,/* pointer to current directory name */ &si, /* pointer to STARTUPINFO */ &ErlProcessInfo /* pointer to PROCESS_INFORMATION */ )) == FALSE) { ShowLastError(); exit_help("Failed to start new node"); } #ifdef _DEBUG fprintf(stderr, "Waiting for Erlang to terminate.\n"); #endif if(MsgWaitForMultipleObjects(1,&ErlProcessInfo.hProcess, FALSE, INFINITE, QS_POSTMESSAGE) == WAIT_OBJECT_0+1){ if(PostThreadMessage(ErlProcessInfo.dwThreadId, WM_USER, (WPARAM) 0, (LPARAM) 0)){ /* Wait 10 seconds for erl process to die, elsee terminate it. */ if(WaitForSingleObject(ErlProcessInfo.hProcess, 10000) != WAIT_OBJECT_0){ TerminateProcess(ErlProcessInfo.hProcess,0); } } else { TerminateProcess(ErlProcessInfo.hProcess,0); } } GetExitCodeProcess(ErlProcessInfo.hProcess, &dwExitCode); #ifdef _DEBUG fprintf(stderr, "Erlang terminated.\n"); #endif free(CommandLine); return(dwExitCode); } /* * Try to make the needed options complete by looking through the data file, * environment variables and registry entries. */ void complete_options(void) { /* Try to find a descent RelDir */ if( !RelDir ) { DWORD sz = 32; while (1) { DWORD nsz; if (RelDir) free(RelDir); RelDir = malloc(sz); if (!RelDir) { fprintf(stderr, "** Error : failed to allocate memory\n"); exit(1); } SetLastError(0); nsz = GetEnvironmentVariable((LPCTSTR) "RELDIR", (LPTSTR) RelDir, sz); if (nsz == 0 && GetLastError() == ERROR_ENVVAR_NOT_FOUND) { free(RelDir); RelDir = NULL; break; } else if (nsz <= sz) break; else sz = nsz; } if (RelDir == NULL) { if(DataFileName){ /* Needs to be absolute for this to work, but we can try... */ read_datafile(); read_registry_keys(); } else { /* Impossible to find all data... */ exit_help("Need either Release directory or an absolute " "datafile name."); } /* Ok, construct our own RelDir from RootDir */ RelDir = (char *) malloc(strlen(RootDir)+strlen(RELEASE_SUBDIR)+1); assert(RelDir); sprintf(RelDir, "%s" RELEASE_SUBDIR, RootDir); } else { read_datafile(); read_registry_keys(); } } else { read_datafile(); read_registry_keys(); } read_bootflags(); #ifdef _DEBUG fprintf(stderr, "RelDir: '%s'\n", RelDir); #endif } BOOL WINAPI LogoffHandlerRoutine( DWORD dwCtrlType ) { if(dwCtrlType == CTRL_LOGOFF_EVENT) { return TRUE; } return FALSE; } int main(void) { DWORD dwExitCode; char *cmdline; /* Make sure a logoff does not distrurb us. */ SetConsoleCtrlHandler(LogoffHandlerRoutine, TRUE); cmdline = GetCommandLine(); assert(cmdline); CommandLineStart = (char *) malloc(strlen(cmdline) + 1); assert(CommandLineStart); strcpy(CommandLineStart,cmdline); split_commandline(); parse_commandline(); complete_options(); /* We now have all the options we need in order to fire up a new node.. */ dwExitCode = start_new_node(); return( (int) dwExitCode ); }