programs/wcmd: Rename to programs/cmd.
diff --git a/programs/cmd/batch.c b/programs/cmd/batch.c
new file mode 100644
index 0000000..8beef16
--- /dev/null
+++ b/programs/cmd/batch.c
@@ -0,0 +1,288 @@
+/*
+ * CMD - Wine-compatible command line interface - batch interface.
+ *
+ * Copyright (C) 1999 D A Pickles
+ *
+ * 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., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
+ */
+
+#include "wcmd.h"
+
+void WCMD_batch_command (char *line);
+
+extern int echo_mode;
+extern char quals[MAX_PATH], param1[MAX_PATH], param2[MAX_PATH];
+extern BATCH_CONTEXT *context;
+extern DWORD errorlevel;
+
+/* msdn specified max for Win XP */
+#define MAXSTRING 8192
+
+/****************************************************************************
+ * WCMD_batch
+ *
+ * Open and execute a batch file.
+ * On entry *command includes the complete command line beginning with the name
+ * of the batch file (if a CALL command was entered the CALL has been removed).
+ * *file is the name of the file, which might not exist and may not have the
+ * .BAT suffix on. Called is 1 for a CALL, 0 otherwise.
+ *
+ * We need to handle recursion correctly, since one batch program might call another.
+ * So parameters for this batch file are held in a BATCH_CONTEXT structure.
+ */
+
+void WCMD_batch (char *file, char *command, int called) {
+
+#define WCMD_BATCH_EXT_SIZE 5
+
+HANDLE h = INVALID_HANDLE_VALUE;
+char string[MAXSTRING];
+char extension_batch[][WCMD_BATCH_EXT_SIZE] = {".bat",".cmd"};
+char extension_exe[WCMD_BATCH_EXT_SIZE] = ".exe";
+unsigned int  i;
+BATCH_CONTEXT *prev_context;
+
+  for(i=0; (i<(sizeof(extension_batch)/WCMD_BATCH_EXT_SIZE)) && 
+           (h == INVALID_HANDLE_VALUE); i++) {
+  strcpy (string, file);
+  CharLower (string);
+    if (strstr (string, extension_batch[i]) == NULL) strcat (string, extension_batch[i]);
+  h = CreateFile (string, GENERIC_READ, FILE_SHARE_READ,
+                  NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
+  }
+  if (h == INVALID_HANDLE_VALUE) {
+    strcpy (string, file);
+    CharLower (string);
+    if (strstr (string, extension_exe) == NULL) strcat (string, extension_exe);
+    h = CreateFile (string, GENERIC_READ, FILE_SHARE_READ,
+                    NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
+    if (h != INVALID_HANDLE_VALUE) {
+      WCMD_run_program (command, 0);
+    } else {
+      SetLastError (ERROR_FILE_NOT_FOUND);
+      WCMD_print_error ();
+    }
+    return;
+  }
+
+/*
+ *	Create a context structure for this batch file.
+ */
+
+  prev_context = context;
+  context = (BATCH_CONTEXT *)LocalAlloc (LMEM_FIXED, sizeof (BATCH_CONTEXT));
+  context -> h = h;
+  context -> command = command;
+  context -> shift_count = 0;
+  context -> prev_context = prev_context;
+
+/*
+ * 	Work through the file line by line. Specific batch commands are processed here,
+ * 	the rest are handled by the main command processor.
+ */
+
+  while (WCMD_fgets (string, sizeof(string), h)) {
+      if (strlen(string) == MAXSTRING -1) {
+          WCMD_output_asis( "Line in Batch processing possibly truncated. Using:\n");
+          WCMD_output_asis( string);
+          WCMD_output_asis( "\n");
+      }
+      if (string[0] != ':') {                      /* Skip over labels */
+          WCMD_batch_command (string);
+      }
+  }
+  CloseHandle (h);
+
+/*
+ *	If invoked by a CALL, we return to the context of our caller. Otherwise return
+ *	to the caller's caller.
+ */
+
+  LocalFree ((HANDLE)context);
+  if ((prev_context != NULL) && (!called)) {
+    CloseHandle (prev_context -> h);
+    context = prev_context -> prev_context;
+    LocalFree ((HANDLE)prev_context);
+  }
+  else {
+    context = prev_context;
+  }
+}
+
+/****************************************************************************
+ * WCMD_batch_command
+ *
+ * Execute one line from a batch file, expanding parameters.
+ */
+
+void WCMD_batch_command (char *line) {
+
+DWORD status;
+char cmd1[MAXSTRING],cmd2[MAXSTRING];
+char *p, *s, *t;
+int i;
+
+  /* Get working version of command line */
+  strcpy(cmd1, line);
+
+  /* Expand environment variables in a batch file %{0-9} first  */
+  /*   Then env vars, and if any left (ie use of undefined vars,*/
+  /*   replace with spaces                                      */
+  /* FIXME: Winnt would replace %1%fred%1 with first parm, then */
+  /*   contents of fred, then the digit 1. Would need to remove */
+  /*   ExpandEnvStrings to achieve this                         */
+
+  /* Replace use of %0...%9 */
+  p = cmd1;
+  while ((p = strchr(p, '%'))) {
+    i = *(p+1) - '0';
+    if ((i >= 0) && (i <= 9)) {
+      s = strdup (p+2);
+      t = WCMD_parameter (context -> command, i + context -> shift_count, NULL);
+      strcpy (p, t);
+      strcat (p, s);
+      free (s);
+    } else {
+      p++;
+    }
+  }
+
+  /* Now replace environment variables */
+  status = ExpandEnvironmentStrings(cmd1, cmd2, sizeof(cmd2));
+  if (!status) {
+    WCMD_print_error ();
+    return;
+  }
+
+  /* In a batch program, unknown variables are replace by nothing */
+  /* so remove any remaining %var%                                */
+  p = cmd2;
+  while ((p = strchr(p, '%'))) {
+    s = strchr(p+1, '%');
+    if (!s) {
+      *p=0x00;
+    } else {
+      t = strdup(s+1);
+      strcpy(p, t);
+      free(t);
+    }
+  }
+
+  /* Show prompt before batch line IF echo is on */
+  if (echo_mode && (line[0] != '@')) {
+    WCMD_show_prompt();
+    WCMD_output_asis ( cmd2);
+    WCMD_output_asis ( "\n");
+  }
+
+  WCMD_process_command (cmd2);
+}
+
+/*******************************************************************
+ * WCMD_parameter - extract a parameter from a command line.
+ *
+ *	Returns the 'n'th space-delimited parameter on the command line (zero-based).
+ *	Parameter is in static storage overwritten on the next call.
+ *	Parameters in quotes (and brackets) are handled.
+ *	Also returns a pointer to the location of the parameter in the command line.
+ */
+
+char *WCMD_parameter (char *s, int n, char **where) {
+
+int i = 0;
+static char param[MAX_PATH];
+char *p;
+
+  p = param;
+  while (TRUE) {
+    switch (*s) {
+      case ' ':
+	s++;
+	break;
+      case '"':
+        if (where != NULL) *where = s;
+	s++;
+	while ((*s != '\0') && (*s != '"')) {
+	  *p++ = *s++;
+	}
+        if (i == n) {
+          *p = '\0';
+          return param;
+        }
+	if (*s == '"') s++;
+          param[0] = '\0';
+          i++;
+        p = param;
+	break;
+      case '(':
+        if (where != NULL) *where = s;
+	s++;
+	while ((*s != '\0') && (*s != ')')) {
+	  *p++ = *s++;
+	}
+        if (i == n) {
+          *p = '\0';
+          return param;
+        }
+	if (*s == ')') s++;
+          param[0] = '\0';
+          i++;
+        p = param;
+	break;
+      case '\0':
+        return param;
+      default:
+        if (where != NULL) *where = s;
+	while ((*s != '\0') && (*s != ' ')) {
+	  *p++ = *s++;
+	}
+        if (i == n) {
+          *p = '\0';
+          return param;
+        }
+          param[0] = '\0';
+          i++;
+        p = param;
+    }
+  }
+}
+
+/****************************************************************************
+ * WCMD_fgets
+ *
+ * Get one line from a batch file. We can't use the native f* functions because
+ * of the filename syntax differences between DOS and Unix. Also need to lose
+ * the LF (or CRLF) from the line.
+ */
+
+char *WCMD_fgets (char *s, int n, HANDLE h) {
+
+DWORD bytes;
+BOOL status;
+char *p;
+
+  p = s;
+  do {
+    status = ReadFile (h, s, 1, &bytes, NULL);
+    if ((status == 0) || ((bytes == 0) && (s == p))) return NULL;
+    if (*s == '\n') bytes = 0;
+    else if (*s != '\r') {
+      s++;
+      n--;
+    }
+    *s = '\0';
+  } while ((bytes == 1) && (n > 1));
+  return p;
+}