/* * CMD - Wine-compatible command line interface. * * Copyright (C) 1999 D A Pickles * Copyright (C) 2007 J Edmeades * * 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 */ #define IDI_ICON1 1 #include <windows.h> #include <windef.h> #ifndef RC_INVOKED #include <string.h> #include <stdlib.h> #include <stdarg.h> #include <stdio.h> #include <ctype.h> #include <wine/unicode.h> /* msdn specified max for Win XP */ #define MAXSTRING 8192 /* Data structure to hold commands delimitors/separators */ typedef enum _CMDdelimiters { CMD_NONE, /* End of line or single & */ CMD_ONFAILURE, /* || */ CMD_ONSUCCESS, /* && */ CMD_PIPE /* Single | */ } CMD_DELIMITERS; /* Data structure to hold commands to be processed */ typedef struct _CMD_LIST { WCHAR *command; /* Command string to execute */ WCHAR *redirects; /* Redirects in place */ struct _CMD_LIST *nextcommand; /* Next command string to execute */ CMD_DELIMITERS prevDelim; /* Previous delimiter */ int bracketDepth;/* How deep bracketing have we got to */ WCHAR pipeFile[MAX_PATH]; /* Where to get input from for pipes */ } CMD_LIST; void WCMD_assoc (const WCHAR *, BOOL); void WCMD_batch (WCHAR *, WCHAR *, BOOL, WCHAR *, HANDLE); void WCMD_call (WCHAR *command); void WCMD_change_tty (void); void WCMD_choice (const WCHAR *); void WCMD_clear_screen (void); void WCMD_color (void); void WCMD_copy (WCHAR *); void WCMD_create_dir (WCHAR *); BOOL WCMD_delete (WCHAR *); void WCMD_directory (WCHAR *); void WCMD_echo (const WCHAR *); void WCMD_endlocal (void); void WCMD_enter_paged_mode(const WCHAR *); void WCMD_exit (CMD_LIST **cmdList); void WCMD_for (WCHAR *, CMD_LIST **cmdList); void WCMD_give_help (const WCHAR *args); void WCMD_goto (CMD_LIST **cmdList); void WCMD_if (WCHAR *, CMD_LIST **cmdList); void WCMD_leave_paged_mode(void); void WCMD_more (WCHAR *); void WCMD_move (void); WCHAR* CDECL WCMD_format_string (const WCHAR *format, ...); void CDECL WCMD_output (const WCHAR *format, ...); void CDECL WCMD_output_stderr (const WCHAR *format, ...); void WCMD_output_asis (const WCHAR *message); void WCMD_output_asis_stderr (const WCHAR *message); void WCMD_pause (void); void WCMD_popd (void); void WCMD_print_error (void); void WCMD_pushd (const WCHAR *args); void WCMD_remove_dir (WCHAR *command); void WCMD_rename (void); void WCMD_run_program (WCHAR *command, BOOL called); void WCMD_setlocal (const WCHAR *args); void WCMD_setshow_date (void); void WCMD_setshow_default (const WCHAR *args); void WCMD_setshow_env (WCHAR *command); void WCMD_setshow_path (const WCHAR *args); void WCMD_setshow_prompt (void); void WCMD_setshow_time (void); void WCMD_shift (const WCHAR *args); void WCMD_start (const WCHAR *args); void WCMD_title (const WCHAR *); void WCMD_type (WCHAR *); void WCMD_verify (const WCHAR *args); void WCMD_version (void); int WCMD_volume (BOOL set_label, const WCHAR *args); static inline BOOL WCMD_is_console_handle(HANDLE h) { return (((DWORD_PTR)h) & 3) == 3; } WCHAR *WCMD_fgets (WCHAR *buf, DWORD n, HANDLE stream); WCHAR *WCMD_parameter (WCHAR *s, int n, WCHAR **start, BOOL raw, BOOL wholecmdline); WCHAR *WCMD_parameter_with_delims (WCHAR *s, int n, WCHAR **start, BOOL raw, BOOL wholecmdline, const WCHAR *delims); WCHAR *WCMD_skip_leading_spaces (WCHAR *string); BOOL WCMD_keyword_ws_found(const WCHAR *keyword, int len, const WCHAR *ptr); void WCMD_HandleTildaModifiers(WCHAR **start, BOOL atExecute); void WCMD_splitpath(const WCHAR* path, WCHAR* drv, WCHAR* dir, WCHAR* name, WCHAR* ext); WCHAR *WCMD_strip_quotes(WCHAR *cmd); WCHAR *WCMD_LoadMessage(UINT id); void WCMD_strsubstW(WCHAR *start, const WCHAR* next, const WCHAR* insert, int len); BOOL WCMD_ReadFile(const HANDLE hIn, WCHAR *intoBuf, const DWORD maxChars, LPDWORD charsRead); WCHAR *WCMD_ReadAndParseLine(const WCHAR *initialcmd, CMD_LIST **output, HANDLE readFrom); CMD_LIST *WCMD_process_commands(CMD_LIST *thisCmd, BOOL oneBracket, BOOL retrycall); void WCMD_free_commands(CMD_LIST *cmds); void WCMD_execute (const WCHAR *orig_command, const WCHAR *redirects, CMD_LIST **cmdList, BOOL retrycall); void *heap_alloc(size_t); static inline BOOL heap_free(void *mem) { return HeapFree(GetProcessHeap(), 0, mem); } static inline WCHAR *heap_strdupW(const WCHAR *str) { WCHAR *ret = NULL; if(str) { size_t size; size = (strlenW(str)+1)*sizeof(WCHAR); ret = heap_alloc(size); memcpy(ret, str, size); } return ret; } static inline BOOL ends_with_backslash( const WCHAR *path ) { return path[0] && path[strlenW(path) - 1] == '\\'; } /* Data structure to hold context when executing batch files */ typedef struct _BATCH_CONTEXT { WCHAR *command; /* The command which invoked the batch file */ HANDLE h; /* Handle to the open batch file */ WCHAR *batchfileW; /* Name of same */ int shift_count[10]; /* Offset in terms of shifts for %0 - %9 */ struct _BATCH_CONTEXT *prev_context; /* Pointer to the previous context block */ BOOL skip_rest; /* Skip the rest of the batch program and exit */ CMD_LIST *toExecute; /* Commands left to be executed */ } BATCH_CONTEXT; /* Data structure to handle building lists during recursive calls */ struct env_stack { struct env_stack *next; union { int stackdepth; /* Only used for pushd and popd */ WCHAR cwd; /* Only used for set/endlocal */ } u; WCHAR *strings; HANDLE batchhandle; /* Used to ensure set/endlocals stay in scope */ BOOL delayedsubst; /* Is delayed substitution in effect */ }; /* Data structure to save setlocal and pushd information */ typedef struct _DIRECTORY_STACK { struct _DIRECTORY_STACK *next; WCHAR *dirName; WCHAR *fileName; } DIRECTORY_STACK; /* Data structure to for loop variables during for body execution, bearing in mind that for loops can be nested */ #define MAX_FOR_VARIABLES 52 #define FOR_VAR_IDX(c) (((c)>='a'&&(c)<='z')?((c)-'a'):\ ((c)>='A'&&(c)<='Z')?(26+(c)-'A'):-1) typedef struct _FOR_CONTEXT { WCHAR *variable[MAX_FOR_VARIABLES]; /* a-z then A-Z */ } FOR_CONTEXT; /* * Global variables quals, param1, param2 contain the current qualifiers * (uppercased and concatenated) and parameters entered, with environment * variables and batch parameters substitution already done. */ extern WCHAR quals[MAX_PATH], param1[MAXSTRING], param2[MAXSTRING]; extern DWORD errorlevel; extern BATCH_CONTEXT *context; extern FOR_CONTEXT forloopcontext; extern BOOL delayedsubst; #endif /* !RC_INVOKED */ /* * Serial nos of builtin commands. These constants must be in step with * the list of strings defined in wcmd.rc, and WCMD_EXIT *must* always be * the last one. * * Yes it *would* be nice to use an enumeration here, but the Resource * Compiler won't accept resource IDs from enumerations :-( */ #define WCMD_CALL 0 #define WCMD_CD 1 #define WCMD_CHDIR 2 #define WCMD_CLS 3 #define WCMD_COPY 4 #define WCMD_CTTY 5 #define WCMD_DATE 6 #define WCMD_DEL 7 #define WCMD_DIR 8 #define WCMD_ECHO 9 #define WCMD_ERASE 10 #define WCMD_FOR 11 #define WCMD_GOTO 12 #define WCMD_HELP 13 #define WCMD_IF 14 #define WCMD_LABEL 15 #define WCMD_MD 16 #define WCMD_MKDIR 17 #define WCMD_MOVE 18 #define WCMD_PATH 19 #define WCMD_PAUSE 20 #define WCMD_PROMPT 21 #define WCMD_REM 22 #define WCMD_REN 23 #define WCMD_RENAME 24 #define WCMD_RD 25 #define WCMD_RMDIR 26 #define WCMD_SET 27 #define WCMD_SHIFT 28 #define WCMD_START 29 #define WCMD_TIME 30 #define WCMD_TITLE 31 #define WCMD_TYPE 32 #define WCMD_VERIFY 33 #define WCMD_VER 34 #define WCMD_VOL 35 #define WCMD_ENDLOCAL 36 #define WCMD_SETLOCAL 37 #define WCMD_PUSHD 38 #define WCMD_POPD 39 #define WCMD_ASSOC 40 #define WCMD_COLOR 41 #define WCMD_FTYPE 42 #define WCMD_MORE 43 #define WCMD_CHOICE 44 /* Must be last in list */ #define WCMD_EXIT 45 /* Some standard messages */ extern const WCHAR newlineW[]; extern const WCHAR spaceW[]; extern const WCHAR nullW[]; extern const WCHAR dotW[]; extern const WCHAR dotdotW[]; extern const WCHAR starW[]; extern const WCHAR slashW[]; extern const WCHAR equalW[]; extern WCHAR anykey[]; extern WCHAR version_string[]; /* Translated messages */ #define WCMD_ALLHELP 1000 #define WCMD_CONFIRM 1001 #define WCMD_YES 1002 #define WCMD_NO 1003 #define WCMD_NOASSOC 1004 #define WCMD_NOFTYPE 1005 #define WCMD_OVERWRITE 1006 #define WCMD_MORESTR 1007 #define WCMD_TRUNCATEDLINE 1008 #define WCMD_NYI 1009 #define WCMD_NOARG 1010 #define WCMD_SYNTAXERR 1011 #define WCMD_FILENOTFOUND 1012 #define WCMD_NOCMDHELP 1013 #define WCMD_NOTARGET 1014 #define WCMD_CURRENTDATE 1015 #define WCMD_CURRENTTIME 1016 #define WCMD_NEWDATE 1017 #define WCMD_NEWTIME 1018 #define WCMD_MISSINGENV 1019 #define WCMD_READFAIL 1020 #define WCMD_CALLINSCRIPT 1021 #define WCMD_ALL 1022 #define WCMD_DELPROMPT 1023 #define WCMD_ECHOPROMPT 1024 #define WCMD_VERIFYPROMPT 1025 #define WCMD_VERIFYERR 1026 #define WCMD_ARGERR 1027 #define WCMD_VOLUMESERIALNO 1028 #define WCMD_VOLUMEPROMPT 1029 #define WCMD_NOPATH 1030 #define WCMD_ANYKEY 1031 #define WCMD_CONSTITLE 1032 #define WCMD_VERSION 1033 #define WCMD_MOREPROMPT 1034 #define WCMD_LINETOOLONG 1035 #define WCMD_VOLUMELABEL 1036 #define WCMD_VOLUMENOLABEL 1037 #define WCMD_YESNO 1038 #define WCMD_YESNOALL 1039 #define WCMD_NO_COMMAND_FOUND 1040 #define WCMD_DIVIDEBYZERO 1041 #define WCMD_NOOPERAND 1042 #define WCMD_NOOPERATOR 1043 #define WCMD_BADPAREN 1044 #define WCMD_BADHEXOCT 1045