/* * WCMD - Wine-compatible command line interface - batch interface. * * (C) 1999 D A Pickles * */ #include "wcmd.h" void WCMD_batch_command (char *line); extern char nyi[]; extern char newline[]; extern char version_string[]; extern int echo_mode; extern char quals[MAX_PATH], param1[MAX_PATH], param2[MAX_PATH]; extern BATCH_CONTEXT *context; extern DWORD errorlevel; /**************************************************************************** * 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) { HANDLE h; char string[MAX_PATH]; BATCH_CONTEXT *prev_context; strcpy (string, file); CharLower (string); if (strstr (string, ".bat") == NULL) strcat (string, ".bat"); h = CreateFile (string, GENERIC_READ, 0, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL); if (h == INVALID_HANDLE_VALUE) { 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 (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[1024],cmd2[1024]; 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 ("%s\n", cmd2); } 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; }