Added Windows-compatible start.exe.

diff --git a/programs/Makefile.in b/programs/Makefile.in
index c3ccd3e..643fdfa 100644
--- a/programs/Makefile.in
+++ b/programs/Makefile.in
@@ -19,6 +19,7 @@
 	regtest \
 	rpcss \
 	rundll32 \
+	start \
 	uninstaller \
 	view \
 	wcmd \
@@ -43,6 +44,7 @@
 	regsvr32 \
 	rpcss \
 	rundll32 \
+	start \
 	uninstaller \
 	wcmd \
 	wineboot \
diff --git a/programs/start/.cvsignore b/programs/start/.cvsignore
new file mode 100644
index 0000000..59ef6ff
--- /dev/null
+++ b/programs/start/.cvsignore
@@ -0,0 +1,4 @@
+Makefile
+rsrc.res
+start.exe.dbg.c
+start.exe.spec.c
diff --git a/programs/start/En.rc b/programs/start/En.rc
new file mode 100644
index 0000000..82e3952
--- /dev/null
+++ b/programs/start/En.rc
@@ -0,0 +1,64 @@
+/*
+ * Start
+ * English Language Support
+ *
+ * Copyright 2003 Dan Kegel
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ * This library 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
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public
+ * License along with this library; if not, write to the Free Software
+ * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
+ */
+
+LANGUAGE LANG_ENGLISH, SUBLANG_ENGLISH_US
+#pragma code_page(850)
+
+STRINGTABLE DISCARDABLE LANGUAGE LANG_ENGLISH, SUBLANG_DEFAULT
+{
+STRING_USAGE,                                   "Start a program, or open a document in the program normally used for files with that suffix. \n\
+Usage: \n\
+start [options] program_filename [...] \n\
+start [options] document_filename \n\
+ \n\
+Options: \n\
+/M[inimized] Start the program minimized. \n\
+/MAX[imized] Start the program maximized. \n\
+/R[estored]  Start the program normally (neither minimized nor maximized). \n\
+/W[ait]      Wait for started program to finish, then exit with its exit code. \n\
+/L           Show end-user license. \n\
+ \n\
+start.exe version 0.2 Copyright (C) 2003, Dan Kegel \n\
+Start comes with ABSOLUTELY NO WARRANTY; for details run with /L option. \n\
+This is free software, and you are welcome to redistribute it \n\
+under certain conditions; run 'start /L' for details. \n\
+"
+
+STRING_LICENSE, "start.exe version 0.2 Copyright (C) 2003, Dan Kegel \n\
+This program is free software; you can redistribute it and/or \n\
+modify it under the terms of the GNU Lesser Public License \n\
+as published by the Free Software Foundation; either version 2.1 \n\
+of the License, or (at your option) any later version. \n\
+ \n\
+This program is distributed in the hope that it will be useful, \n\
+but WITHOUT ANY WARRANTY; without even the implied warranty of \n\
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the \n\
+GNU Lesser Public License for more details. \n\
+ \n\
+You should have received a copy of the GNU Lesser Public License \n\
+along with this program; if not, write to the Free Software \n\
+Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA. \n\
+ \n\
+See the COPYING.LIB file for license information. \n\
+"
+
+STRING_EXECFAIL "Application could not be started, or no application associated with the specified file.\nShellExecuteEx failed"
+}
diff --git a/programs/start/Makefile.in b/programs/start/Makefile.in
new file mode 100644
index 0000000..2a9291d
--- /dev/null
+++ b/programs/start/Makefile.in
@@ -0,0 +1,15 @@
+TOPSRCDIR = @top_srcdir@
+TOPOBJDIR = ../..
+SRCDIR    = @srcdir@
+VPATH     = @srcdir@
+MODULE    = start.exe
+APPMODE   = cui
+IMPORTS   = shell32 user32 kernel32
+
+C_SRCS = start.c
+
+RC_SRCS = rsrc.rc
+
+@MAKE_PROG_RULES@
+
+### Dependencies:
diff --git a/programs/start/resources.h b/programs/start/resources.h
new file mode 100644
index 0000000..c81296a
--- /dev/null
+++ b/programs/start/resources.h
@@ -0,0 +1,23 @@
+/*
+ * Resource id definitions
+ *
+ * Copyright 2003 Dan Kegel
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 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
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public
+ * License along with this program; if not, write to the Free Software
+ * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
+ */
+
+#define STRING_USAGE 101
+#define STRING_LICENSE 102
+#define STRING_EXECFAIL 103
diff --git a/programs/start/rsrc.rc b/programs/start/rsrc.rc
new file mode 100644
index 0000000..8a77553
--- /dev/null
+++ b/programs/start/rsrc.rc
@@ -0,0 +1,24 @@
+/*
+ * Copyright 2003 Dan Kegel
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ * This library 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
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public
+ * License along with this library; if not, write to the Free Software
+ * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
+ */
+
+#include <windows.h>
+#include "resources.h"
+
+LANGUAGE LANG_NEUTRAL, SUBLANG_NEUTRAL
+
+#include "En.rc"
diff --git a/programs/start/start.c b/programs/start/start.c
new file mode 100644
index 0000000..faba079
--- /dev/null
+++ b/programs/start/start.c
@@ -0,0 +1,190 @@
+/*
+ * Start a program using ShellExecuteEx, optionally wait for it to finish
+ * Compatible with Microsoft's "c:\windows\command\start.exe"
+ *
+ * Copyright 2003 Dan Kegel
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 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
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public
+ * License along with this program; if not, write to the Free Software
+ * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
+ */
+
+#include <windows.h>
+#include <winuser.h>
+#include <stdio.h>
+#include <stdlib.h>
+#include <shlobj.h>
+
+#include "resources.h"
+
+/**
+ Output given message to stdout without formatting.
+*/
+static void output(const char *message)
+{
+	DWORD count;
+	WriteFile(GetStdHandle(STD_OUTPUT_HANDLE), message, strlen(message), &count, NULL);
+}
+
+/**
+ Output given message,
+ followed by ": ",
+ followed by description of given GetLastError() value to stdout,
+ followed by a trailing newline,
+ then terminate.
+*/
+static void fatal_error(const char *msg, DWORD error_code)
+{
+	LPVOID lpMsgBuf;
+	int status;
+
+	output(msg);
+	output(": ");
+	status = FormatMessage(FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM, NULL, error_code, 0, (LPTSTR) & lpMsgBuf, 0, NULL);
+	if (!status) {
+		output("FormatMessage failed\n");
+	} else {
+		output(lpMsgBuf);
+		LocalFree((HLOCAL) lpMsgBuf);
+		output("\n");
+	}
+	ExitProcess(1);
+}
+
+/**
+ Output given message from string table,
+ followed by ": ",
+ followed by description of given GetLastError() value to stdout,
+ followed by a trailing newline,
+ then terminate.
+*/
+static void fatal_string_error(int which, DWORD error_code)
+{
+	char msg[2048];
+
+	if (!LoadString(GetModuleHandle(NULL), which, 
+					msg, sizeof(msg)))
+		fatal_error("LoadString failed", GetLastError());
+
+	fatal_error(msg, error_code);
+}
+	
+static void fatal_string(int which)
+{
+	char msg[2048];
+
+	if (!LoadString(GetModuleHandle(NULL), which, 
+					msg, sizeof(msg)))
+		fatal_error("LoadString failed", GetLastError());
+
+	output(msg);
+	ExitProcess(1);
+}
+
+static void usage()
+{
+	fatal_string(STRING_USAGE);
+}
+
+static void license()
+{
+	fatal_string(STRING_LICENSE);
+}
+
+int main(int argc, char *argv[])
+{
+	char arguments[MAX_PATH];
+	char *p;
+	SHELLEXECUTEINFO sei;
+	int argi;
+
+	memset(&sei, 0, sizeof(sei));
+	sei.cbSize = sizeof(sei);
+	sei.lpVerb = "open";
+	sei.nShow = SW_SHOWNORMAL;
+	/* Dunno what these mean, but it looks like winMe's start uses them */
+	sei.fMask = SEE_MASK_FLAG_DDEWAIT|SEE_MASK_FLAG_NO_UI;
+
+	/* Canonical Microsoft commandline flag processing:
+	 * flags start with /, are case insensitive,
+	 * and may be run together in same word.
+	 */
+	for (argi=1; argi<argc; argi++) {
+		int ci;
+
+		if (argv[argi][0] != '/')
+			break;
+
+		/* Handle all options in this word */
+		for (ci=0; argv[argi][ci]; ) {
+			/* Skip slash */
+			ci++;
+			switch(argv[argi][ci]) {
+			case 'l':
+			case 'L':
+				license();
+				break;	/* notreached */
+			case 'm':
+			case 'M':
+				if (argv[argi][ci+1] == 'a' || argv[argi][ci+1] == 'A')
+					sei.nShow = SW_SHOWMAXIMIZED;
+				else
+					sei.nShow = SW_SHOWMINIMIZED;
+				break;
+			case 'r':
+			case 'R':
+				/* sei.nShow = SW_SHOWNORMAL; */
+				break;
+			case 'w':
+			case 'W':
+				sei.fMask |= SEE_MASK_NOCLOSEPROCESS;
+				break;
+			default:
+				printf("Option '%s' not recognized\n", argv[argi]+ci-1);
+				usage();
+			}
+			/* Skip to next slash */
+			while (argv[argi][ci] && (argv[argi][ci] != '/'))
+				ci++;
+		}
+	}
+
+	if (argi == argc)
+		usage();
+
+	sei.lpFile = argv[argi++];
+
+	/* FIXME - prone to overflow */
+	arguments[0] = 0;
+	for (p = arguments; argi < argc; argi++)
+		p += sprintf(p, " %s", argv[argi]);
+
+	sei.lpParameters = arguments;
+
+	if (!ShellExecuteEx(&sei))
+	    	fatal_string_error(STRING_EXECFAIL, GetLastError());
+
+	if (sei.fMask & SEE_MASK_NOCLOSEPROCESS) {
+		DWORD exitcode;
+		DWORD waitcode;
+		waitcode = WaitForSingleObject(sei.hProcess, INFINITE);
+		if (waitcode)
+			fatal_error("WaitForSingleObject", GetLastError());
+		if (!GetExitCodeProcess(sei.hProcess, &exitcode))
+			fatal_error("GetExitCodeProcess", GetLastError());
+		/* fixme: haven't tested whether exit code works properly */
+		ExitProcess(exitcode);
+	}
+
+	ExitProcess(0);
+}