aboutsummaryrefslogblamecommitdiffstats
path: root/erts/etc/win32/erl.c
blob: d3411539664f9b7567caa06f84ff0de227ba4bd7 (plain) (tree)


























































































































































































































































































                                                                                    
/*
 * %CopyrightBegin%
 * 
 * Copyright Ericsson AB 2003-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%
 */
#pragma comment(linker,"/manifestdependency:\"type='win32' "\
		"name='Microsoft.Windows.Common-Controls' "\
		"version='6.0.0.0' processorArchitecture='*' "\
		"publicKeyToken='6595b64144ccf1df' language='*'\"")
#include <windows.h>
#include <stdio.h>
#include <stdlib.h>
#include "init_file.h"

typedef int ErlexecFunction(int, char **, HANDLE, int); 

#define INI_FILENAME "erl.ini"
#define INI_SECTION "erlang"
#define ERLEXEC_BASENAME "erlexec.dll"

static void get_parameters(void);
static void error(char* format, ...);

static char *erlexec_name;
static char *erlexec_dir;

#ifdef WIN32_WERL
#define WERL 1
int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance,
		    PSTR szCmdLine, int iCmdShow)
{
    int argc = __argc;
    char **argv = __argv;
#else
#define WERL 0
int main(int argc, char **argv)
{
#endif
  HANDLE erlexec_handle; /* Instance */
  ErlexecFunction *win_erlexec;
  char *path = malloc(100);
  char *npath;
  int pathlen;

  get_parameters();

  if ((pathlen = GetEnvironmentVariable("PATH",path,100)) == 0) {
    error("No PATH variable (!)");
  } else if (pathlen > 100) {
    path = realloc(path,pathlen);
    GetEnvironmentVariable("PATH",path,pathlen);
  }
  npath = malloc(strlen(path) + strlen(erlexec_dir) + 2);
  sprintf(npath,"%s;%s",erlexec_dir,path);
  SetEnvironmentVariable("PATH",npath);

  if ((erlexec_handle = LoadLibrary(erlexec_name)) == NULL) {
    error("Could not load module %s.",erlexec_name);
  }

  if ((win_erlexec = (ErlexecFunction *) 
       GetProcAddress(erlexec_handle,"win_erlexec")) == NULL) {
    error("Could not find entry point \"win_erlexec\" in %s.", erlexec_name);
  }

  return (*win_erlexec)(argc,argv,erlexec_handle,WERL);
  
} 


static char *replace_filename(char *path, char *new_base) 
{
    int plen = strlen(path);
    char *res = malloc((plen+strlen(new_base)+1)*sizeof(char));
    char *p;

    strcpy(res,path);
    for (p = res+plen-1 ;p >= res && *p != '\\'; --p)
        ;
    *(p+1) ='\0';
    strcat(res,new_base);
    return res;
}

static char *do_lookup_in_section(InitSection *inis, char *name, 
				  char *section, char *filename)
{
    char *p = lookup_init_entry(inis, name);

    if (p == NULL) {
	error("Could not find key %s in section %s of file %s",
	      name,section,filename);
    }
    return _strdup(p);
}

static void copy_latest_vsn(char *latest_vsn, char *next_vsn) 
{
    /* Copy */
    char *lp;
    char *np;
    /* Find vsn */
    for (lp = next_vsn+strlen(next_vsn)-1 ;lp >= next_vsn && *lp != '\\'; --lp)
        ;
    /* lp =+ length("erts-"); */
    for (np = next_vsn+strlen(next_vsn)-1 ;np >= next_vsn && *np != '\\'; --np)
        ;
    /* np =+ length("erts-"); */
    
    for (; lp && np; ++lp, ++np) {
	if (*lp == *np) {
	    continue;
	}	
	if (*np == '.' || *np == '\0' || *np <= *lp) {
	/* */
	    return;
	}
	if (*lp == '.' || *lp == '\0') {
	    strcpy(latest_vsn, next_vsn);
	    return;
	}
    }
    return;
}

static char *find_erlexec_dir2(char *install_dir) 
{
    /* List install dir and look for latest erts-vsn */

    HANDLE dir_handle;	        /* Handle to directory. */
    char wildcard[MAX_PATH];	/* Wildcard to search for. */
    WIN32_FIND_DATA find_data;	/* Data found by FindFirstFile() or FindNext(). */
    char latest_vsn[MAX_PATH];

    /* Setup wildcard */
    int length = strlen(install_dir);
    char *p;

    if (length+3 >= MAX_PATH) {
	error("Cannot find erlexec.exe");
    }

    strcpy(wildcard, install_dir);
    p = wildcard+length-1;
    if (*p != '/' && *p != '\\')
	*++p = '\\';
    strcpy(++p, "erts-*");

    /* Find first dir */
    dir_handle = FindFirstFile(wildcard, &find_data);
    if (dir_handle == INVALID_HANDLE_VALUE) {
	/* No erts-vsn found*/
	return NULL;
    }	
    strcpy(latest_vsn, find_data.cFileName);

    /* Find the rest */
    while(FindNextFile(dir_handle, &find_data)) {
	copy_latest_vsn(latest_vsn, find_data.cFileName);
    }
    
    FindClose(dir_handle);

    p = malloc((strlen(install_dir)+1+strlen(latest_vsn)+4+1)*sizeof(char));

    strcpy(p,install_dir);
    strcat(p,"\\");
    strcat(p,latest_vsn);
    strcat(p,"\\bin");
    return p;
}

static char *find_erlexec_dir(char *erlpath) 
{
    /* Assume that the path to erl is absolute and
     * that it is not a symbolic link*/
    
    char *dir =_strdup(erlpath);
    char *p;
    char *p2;
    
    /* Chop of base name*/
    for (p = dir+strlen(dir)-1 ;p >= dir && *p != '\\'; --p)
        ;
    *p ='\0';
    p--;

    /* Check if dir path is like ...\install_dir\erts-vsn\bin */
    for (;p >= dir && *p != '\\'; --p)
        ;
    p--;
    for (p2 = p;p2 >= dir && *p2 != '\\'; --p2)
        ;
    p2++;
    if (strncmp(p2, "erts-", strlen("erts-")) == 0) {
	p = _strdup(dir);
	free(dir);
	return p;
    }

    /* Assume that dir path is like ...\install_dir\bin */
    *++p ='\0'; /* chop off bin dir */

    p = find_erlexec_dir2(dir);
    free(dir);
    if (p == NULL) {
	error("Cannot find erlexec.exe");
    } else {
	return p;
    }
}

static void get_parameters(void)
{
    char buffer[MAX_PATH];
    char *ini_filename;
    HANDLE module = GetModuleHandle(NULL);
    InitFile *inif;
    InitSection *inis;
    char *bindir;

    if (module = NULL) {
        error("Cannot GetModuleHandle()");
    }

    if (GetModuleFileName(module,buffer,MAX_PATH) == 0) {
        error("Could not GetModuleFileName");
    }

    ini_filename = replace_filename(buffer,INI_FILENAME);

    if ((inif = load_init_file(ini_filename)) == NULL) {
	erlexec_dir = find_erlexec_dir(ini_filename);
	SetEnvironmentVariable("ERLEXEC_DIR", erlexec_dir);
    } else {

      if ((inis = lookup_init_section(inif,INI_SECTION)) == NULL) {
	error("Could not find section %s in init file %s",
	      INI_SECTION, ini_filename);
      }
      
      erlexec_dir = do_lookup_in_section(inis, "Bindir", INI_SECTION, ini_filename);
      free_init_file(inif);      
    }
    
    erlexec_name = malloc(strlen(erlexec_dir) + strlen(ERLEXEC_BASENAME) + 2);
    strcpy(erlexec_name,erlexec_dir);
    strcat(erlexec_name, "\\" ERLEXEC_BASENAME);
    
    free(ini_filename);
}


static void error(char* format, ...)
{
    char sbuf[2048];
    va_list ap;

    va_start(ap, format);
    vsprintf(sbuf, format, ap);
    va_end(ap);

#ifndef WIN32_WERL 
	fprintf(stderr, "%s\n", sbuf);
#else
	MessageBox(NULL, sbuf, "Werl", MB_OK|MB_ICONERROR);
#endif
    exit(1);
}