| /* |
| * WCMD - Wine-compatible command line interface - batch interface. |
| * |
| * (C) 1999 D A Pickles |
| * |
| */ |
| |
| |
| #include "wcmd.h" |
| |
| void WCMD_batch_command (HANDLE h, char *command); |
| char *WCMD_parameter (char *s, int n); |
| BOOL WCMD_go_to (HANDLE h, char *label); |
| |
| extern HANDLE STDin, STDout; |
| 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]; |
| |
| |
| |
| /**************************************************************************** |
| * 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. |
| * |
| * We need to handle recursion correctly, since one batch program might call another. |
| */ |
| |
| void WCMD_batch (char *file, char *command) { |
| |
| HANDLE h; |
| char string[MAX_PATH]; |
| int n; |
| |
| 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, 0); |
| if (h == INVALID_HANDLE_VALUE) { |
| WCMD_output ("File %s not found\n", string); |
| return; |
| } |
| |
| /* |
| * 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)) { |
| n = strlen (string); |
| if (string[n-1] == '\n') string[n-1] = '\0'; |
| if (string[n-2] == '\r') string[n-2] = '\0'; /* Under Windoze we get CRLF! */ |
| WCMD_batch_command (h, string); |
| } |
| CloseHandle (h); |
| } |
| |
| /**************************************************************************** |
| * WCMD_batch_command |
| * |
| * Execute one line from a batch file. |
| */ |
| |
| void WCMD_batch_command (HANDLE h, char *command) { |
| |
| DWORD status; |
| char cmd[1024]; |
| |
| if (echo_mode && (command[0] != '@')) WCMD_output ("%s", command); |
| status = ExpandEnvironmentStrings (command, cmd, sizeof(cmd)); |
| if (!status) { |
| WCMD_print_error (); |
| return; |
| } |
| WCMD_process_command (cmd); |
| } |
| |
| /**************************************************************************** |
| * WCMD_go_to |
| * |
| * Batch file jump instruction. Not the most efficient algorithm ;-) |
| * Returns FALSE if the specified label cannot be found - the file pointer is |
| * then at EOF. |
| */ |
| |
| BOOL WCMD_go_to (HANDLE h, char *label) { |
| |
| char string[MAX_PATH]; |
| |
| SetFilePointer (h, 0, NULL, FILE_BEGIN); |
| while (WCMD_fgets (string, sizeof(string), h)) { |
| if ((string[0] == ':') && (strcmp (&string[1], label) == 0)) return TRUE; |
| } |
| return FALSE; |
| } |
| |
| /******************************************************************* |
| * WCMD_parameter - extract a parameter from a command line. |
| * |
| * Returns the 'n'th space-delimited parameter on the command line. |
| * Parameter is in static storage overwritten on the next call. |
| * Parameters in quotes are handled. |
| */ |
| |
| char *WCMD_parameter (char *s, int n) { |
| |
| int i = -1; |
| static char param[MAX_PATH]; |
| char *p; |
| |
| p = param; |
| while (TRUE) { |
| switch (*s) { |
| case ' ': |
| s++; |
| break; |
| case '"': |
| s++; |
| while ((*s != '\0') && (*s != '"')) { |
| *p++ = *s++; |
| } |
| if (i == n) { |
| *p = '\0'; |
| return param; |
| } |
| else { |
| param[0] = '\0'; |
| i++; |
| } |
| if (*s == '"') s++; |
| break; |
| case '\0': |
| return param; |
| default: |
| while ((*s != '\0') && (*s != ' ')) { |
| *p++ = *s++; |
| } |
| if (i == n) { |
| *p = '\0'; |
| return param; |
| } |
| else { |
| param[0] = '\0'; |
| i++; |
| } |
| } |
| } |
| } |
| |
| /**************************************************************************** |
| * 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. |
| */ |
| |
| 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)) return NULL; |
| if (*s == '\n') bytes = 0; |
| *++s = '\0'; |
| n--; |
| } while ((bytes == 1) && (n > 1)); |
| return p; |
| } |