blob: d887c4f480abeec10e868ebcae6c186ac1bd807e [file] [log] [blame]
Dave Pickles74f440e1999-06-06 15:24:04 +00001/*
Dan Kegel39857442006-09-03 21:13:08 -07002 * CMD - Wine-compatible command line interface - built-in functions.
Dave Pickles74f440e1999-06-06 15:24:04 +00003 *
Alexandre Julliard0799c1a2002-03-09 23:29:33 +00004 * Copyright (C) 1999 D A Pickles
Dave Pickles74f440e1999-06-06 15:24:04 +00005 *
Alexandre Julliard0799c1a2002-03-09 23:29:33 +00006 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2.1 of the License, or (at your option) any later version.
10 *
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
15 *
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, write to the Free Software
Jonathan Ernst360a3f92006-05-18 14:49:52 +020018 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
Alexandre Julliard0799c1a2002-03-09 23:29:33 +000019 */
20
21/*
22 * NOTES:
Dave Pickles74f440e1999-06-06 15:24:04 +000023 * On entry to each function, global variables quals, param1, param2 contain
24 * the qualifiers (uppercased and concatenated) and parameters entered, with
25 * environment-variable and batch parameter substitution already done.
26 */
27
28/*
29 * FIXME:
Dave Pickles036a9f71999-07-10 11:36:45 +000030 * - No support for pipes, shell parameters
Dave Pickles74f440e1999-06-06 15:24:04 +000031 * - Lots of functionality missing from builtins
32 * - Messages etc need international support
33 */
34
Mike McCormack7d665672006-01-16 20:41:34 +010035#define WIN32_LEAN_AND_MEAN
36
Dave Pickles74f440e1999-06-06 15:24:04 +000037#include "wcmd.h"
Jason Edmeades69194ce2007-02-26 23:04:40 +000038#include <shellapi.h>
Jason Edmeades0efa91d2007-03-04 22:33:27 +000039#include "wine/debug.h"
40
41WINE_DEFAULT_DEBUG_CHANNEL(cmd);
Dave Pickles74f440e1999-06-06 15:24:04 +000042
Jason Edmeades54d890c2007-06-15 20:59:28 +010043static void WCMD_part_execute(CMD_LIST **commands, WCHAR *firstcmd, WCHAR *variable,
44 WCHAR *value, BOOL isIF, BOOL conditionTRUE);
Dave Pickles036a9f71999-07-10 11:36:45 +000045
Mike McCormack9643b792004-03-22 22:56:58 +000046struct env_stack *saved_environment;
Jason Edmeades365f86f2007-02-23 22:17:28 +000047struct env_stack *pushd_directories;
Mike McCormack9643b792004-03-22 22:56:58 +000048
Dave Pickles74f440e1999-06-06 15:24:04 +000049extern HINSTANCE hinst;
Jason Edmeadesafe4d802007-06-03 22:07:43 +010050extern WCHAR inbuilt[][10];
Jason Edmeadese37463f2007-03-08 00:37:44 +000051extern int echo_mode, verify_mode, defaultColor;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +010052extern WCHAR quals[MAX_PATH], param1[MAX_PATH], param2[MAX_PATH];
Dave Pickles5f8f4f71999-06-26 10:24:08 +000053extern BATCH_CONTEXT *context;
Dave Picklesebecf502000-08-01 22:02:18 +000054extern DWORD errorlevel;
Dave Pickles74f440e1999-06-06 15:24:04 +000055
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +010056static const WCHAR dotW[] = {'.','\0'};
57static const WCHAR dotdotW[] = {'.','.','\0'};
58static const WCHAR slashW[] = {'\\','\0'};
59static const WCHAR starW[] = {'*','\0'};
60static const WCHAR equalW[] = {'=','\0'};
61static const WCHAR fslashW[] = {'/','\0'};
62static const WCHAR onW[] = {'O','N','\0'};
63static const WCHAR offW[] = {'O','F','F','\0'};
64static const WCHAR parmY[] = {'/','Y','\0'};
65static const WCHAR parmNoY[] = {'/','-','Y','\0'};
66static const WCHAR nullW[] = {'\0'};
Dave Pickles74f440e1999-06-06 15:24:04 +000067
68/****************************************************************************
69 * WCMD_clear_screen
70 *
71 * Clear the terminal screen.
72 */
73
Eric Pouechb2f079b2003-02-27 01:41:21 +000074void WCMD_clear_screen (void) {
Dave Pickles74f440e1999-06-06 15:24:04 +000075
Jason Edmeadesa4c214e2002-04-29 17:12:57 +000076 /* Emulate by filling the screen from the top left to bottom right with
77 spaces, then moving the cursor to the top left afterwards */
Jason Edmeadesa4c214e2002-04-29 17:12:57 +000078 CONSOLE_SCREEN_BUFFER_INFO consoleInfo;
79 HANDLE hStdOut = GetStdHandle(STD_OUTPUT_HANDLE);
Dave Pickles74f440e1999-06-06 15:24:04 +000080
Eric Pouechb2f079b2003-02-27 01:41:21 +000081 if (GetConsoleScreenBufferInfo(hStdOut, &consoleInfo))
82 {
83 COORD topLeft;
Mike McCormack516a5772005-08-19 10:04:03 +000084 DWORD screenSize;
85
Eric Pouechb2f079b2003-02-27 01:41:21 +000086 screenSize = consoleInfo.dwSize.X * (consoleInfo.dwSize.Y + 1);
Jason Edmeadesa4c214e2002-04-29 17:12:57 +000087
Eric Pouechb2f079b2003-02-27 01:41:21 +000088 topLeft.X = 0;
89 topLeft.Y = 0;
90 FillConsoleOutputCharacter(hStdOut, ' ', screenSize, topLeft, &screenSize);
91 SetConsoleCursorPosition(hStdOut, topLeft);
92 }
Dave Pickles74f440e1999-06-06 15:24:04 +000093}
94
95/****************************************************************************
96 * WCMD_change_tty
97 *
98 * Change the default i/o device (ie redirect STDin/STDout).
99 */
100
Eric Pouechb2f079b2003-02-27 01:41:21 +0000101void WCMD_change_tty (void) {
Dave Pickles74f440e1999-06-06 15:24:04 +0000102
Jason Edmeades5cc492c2007-06-03 22:07:39 +0100103 WCMD_output (WCMD_LoadMessage(WCMD_NYI));
Dave Pickles74f440e1999-06-06 15:24:04 +0000104
105}
106
107/****************************************************************************
108 * WCMD_copy
109 *
110 * Copy a file or wildcarded set.
Jason Edmeadesa766ba02007-07-27 21:43:37 +0100111 * FIXME: Add support for a+b+c type syntax
Dave Pickles74f440e1999-06-06 15:24:04 +0000112 */
113
Eric Pouechb2f079b2003-02-27 01:41:21 +0000114void WCMD_copy (void) {
Dave Pickles74f440e1999-06-06 15:24:04 +0000115
Alexandre Julliardabfe1052007-03-01 12:43:19 +0100116 WIN32_FIND_DATA fd;
117 HANDLE hff;
118 BOOL force, status;
Jason Edmeadesa766ba02007-07-27 21:43:37 +0100119 WCHAR outpath[MAX_PATH], srcpath[MAX_PATH], copycmd[3];
Alexandre Julliardabfe1052007-03-01 12:43:19 +0100120 DWORD len;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100121 static const WCHAR copyCmdW[] = {'C','O','P','Y','C','M','D','\0'};
Jason Edmeadesa766ba02007-07-27 21:43:37 +0100122 BOOL copyToDir = FALSE;
123 BOOL copyFromDir = FALSE;
124 WCHAR srcspec[MAX_PATH];
125 DWORD attribs;
126 WCHAR drive[10];
127 WCHAR dir[MAX_PATH];
128 WCHAR fname[MAX_PATH];
129 WCHAR ext[MAX_PATH];
Dave Pickles74f440e1999-06-06 15:24:04 +0000130
Markus Amsler765ff5d2006-10-31 21:11:28 +0100131 if (param1[0] == 0x00) {
Jason Edmeades5cc492c2007-06-03 22:07:39 +0100132 WCMD_output (WCMD_LoadMessage(WCMD_NOARG));
Markus Amsler765ff5d2006-10-31 21:11:28 +0100133 return;
134 }
135
Jason Edmeadesa766ba02007-07-27 21:43:37 +0100136 /* Convert source into full spec */
137 WINE_TRACE("Copy source (supplied): '%s'\n", wine_dbgstr_w(param1));
138 GetFullPathName (param1, sizeof(srcpath)/sizeof(WCHAR), srcpath, NULL);
139 if (srcpath[strlenW(srcpath) - 1] == '\\')
140 srcpath[strlenW(srcpath) - 1] = '\0';
141
142 if ((strchrW(srcpath,'*') == NULL) && (strchrW(srcpath,'?') == NULL)) {
143 attribs = GetFileAttributes(srcpath);
144 } else {
145 attribs = 0;
146 }
147 strcpyW(srcspec, srcpath);
148
149 /* If a directory, then add \* on the end when searching */
150 if (attribs & FILE_ATTRIBUTE_DIRECTORY) {
151 strcatW(srcpath, slashW);
152 copyFromDir = TRUE;
153 strcatW(srcspec, slashW);
154 strcatW(srcspec, starW);
155 } else {
156 WCMD_splitpath(srcpath, drive, dir, fname, ext);
157 strcpyW(srcpath, drive);
158 strcatW(srcpath, dir);
Dave Pickles74f440e1999-06-06 15:24:04 +0000159 }
Jason Edmeades13c51172002-04-20 20:54:38 +0000160
Jason Edmeadesa766ba02007-07-27 21:43:37 +0100161 WINE_TRACE("Copy source (calculated): path: '%s'\n", wine_dbgstr_w(srcpath));
162
Jason Edmeades13c51172002-04-20 20:54:38 +0000163 /* If no destination supplied, assume current directory */
Jason Edmeadesa766ba02007-07-27 21:43:37 +0100164 WINE_TRACE("Copy destination (supplied): '%s'\n", wine_dbgstr_w(param2));
Jason Edmeades13c51172002-04-20 20:54:38 +0000165 if (param2[0] == 0x00) {
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100166 strcpyW(param2, dotW);
Jason Edmeades13c51172002-04-20 20:54:38 +0000167 }
168
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100169 GetFullPathName (param2, sizeof(outpath)/sizeof(WCHAR), outpath, NULL);
170 if (outpath[strlenW(outpath) - 1] == '\\')
171 outpath[strlenW(outpath) - 1] = '\0';
Jason Edmeadesa766ba02007-07-27 21:43:37 +0100172 attribs = GetFileAttributes(outpath);
173 if (attribs & FILE_ATTRIBUTE_DIRECTORY) {
174 strcatW (outpath, slashW);
175 copyToDir = TRUE;
Dave Picklesebecf502000-08-01 22:02:18 +0000176 }
Jason Edmeadesa766ba02007-07-27 21:43:37 +0100177 WINE_TRACE("Copy destination (calculated): '%s'(%d)\n",
178 wine_dbgstr_w(outpath), copyToDir);
Jason Edmeades13c51172002-04-20 20:54:38 +0000179
Alexander Farberf6ec4412007-02-28 14:49:46 +0100180 /* /-Y has the highest priority, then /Y and finally the COPYCMD env. variable */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100181 if (strstrW (quals, parmNoY))
Alexander Farberf6ec4412007-02-28 14:49:46 +0100182 force = FALSE;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100183 else if (strstrW (quals, parmY))
Alexander Farberf6ec4412007-02-28 14:49:46 +0100184 force = TRUE;
185 else {
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100186 len = GetEnvironmentVariable (copyCmdW, copycmd, sizeof(copycmd)/sizeof(WCHAR));
187 force = (len && len < (sizeof(copycmd)/sizeof(WCHAR)) && ! lstrcmpiW (copycmd, parmY));
Alexander Farberf6ec4412007-02-28 14:49:46 +0100188 }
189
Jason Edmeadesa766ba02007-07-27 21:43:37 +0100190 /* Loop through all source files */
191 WINE_TRACE("Searching for: '%s'\n", wine_dbgstr_w(srcspec));
192 hff = FindFirstFile (srcspec, &fd);
193 if (hff != INVALID_HANDLE_VALUE) {
194 do {
195 WCHAR outname[MAX_PATH];
196 WCHAR srcname[MAX_PATH];
197 BOOL overwrite = force;
Jason Edmeades5cc492c2007-06-03 22:07:39 +0100198
Jason Edmeadesa766ba02007-07-27 21:43:37 +0100199 /* Destination is either supplied filename, or source name in
200 supplied destination directory */
201 strcpyW(outname, outpath);
202 if (copyToDir) strcatW(outname, fd.cFileName);
203 strcpyW(srcname, srcpath);
204 strcatW(srcname, fd.cFileName);
205
206 WINE_TRACE("Copying from : '%s'\n", wine_dbgstr_w(srcname));
207 WINE_TRACE("Copying to : '%s'\n", wine_dbgstr_w(outname));
208
209 /* Skip . and .., and directories */
210 if (fd.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) {
211 overwrite = FALSE;
212 WINE_TRACE("Skipping directories\n");
213 }
214
215 /* Prompt before overwriting */
216 else if (!overwrite) {
217 attribs = GetFileAttributes(outname);
218 if (attribs != INVALID_FILE_ATTRIBUTES) {
219 WCHAR buffer[MAXSTRING];
220 wsprintf(buffer, WCMD_LoadMessage(WCMD_OVERWRITE), outname);
221 overwrite = WCMD_ask_confirm(buffer, FALSE, NULL);
222 }
223 else overwrite = TRUE;
224 }
225
226 /* Do the copy as appropriate */
227 if (overwrite) {
228 status = CopyFile (srcname, outname, FALSE);
229 if (!status) WCMD_print_error ();
230 }
231
232 } while (FindNextFile(hff, &fd) != 0);
Dave Pickles74f440e1999-06-06 15:24:04 +0000233 FindClose (hff);
Jason Edmeadesa766ba02007-07-27 21:43:37 +0100234 } else {
235 status = ERROR_FILE_NOT_FOUND;
236 WCMD_print_error ();
Dave Pickles74f440e1999-06-06 15:24:04 +0000237 }
238}
239
240/****************************************************************************
241 * WCMD_create_dir
242 *
243 * Create a directory.
Aric Stewart1d5adff2005-12-03 18:02:15 +0100244 *
245 * this works recursivly. so mkdir dir1\dir2\dir3 will create dir1 and dir2 if
246 * they do not already exist.
Dave Pickles74f440e1999-06-06 15:24:04 +0000247 */
248
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100249static BOOL create_full_path(WCHAR* path)
Aric Stewart1d5adff2005-12-03 18:02:15 +0100250{
251 int len;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100252 WCHAR *new_path;
Aric Stewart1d5adff2005-12-03 18:02:15 +0100253 BOOL ret = TRUE;
254
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100255 new_path = HeapAlloc(GetProcessHeap(),0,(strlenW(path) * sizeof(WCHAR))+1);
256 strcpyW(new_path,path);
Aric Stewart1d5adff2005-12-03 18:02:15 +0100257
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100258 while ((len = strlenW(new_path)) && new_path[len - 1] == '\\')
Aric Stewart1d5adff2005-12-03 18:02:15 +0100259 new_path[len - 1] = 0;
260
261 while (!CreateDirectory(new_path,NULL))
262 {
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100263 WCHAR *slash;
Aric Stewart1d5adff2005-12-03 18:02:15 +0100264 DWORD last_error = GetLastError();
265 if (last_error == ERROR_ALREADY_EXISTS)
266 break;
267
268 if (last_error != ERROR_PATH_NOT_FOUND)
269 {
270 ret = FALSE;
271 break;
272 }
273
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100274 if (!(slash = strrchrW(new_path,'\\')) && ! (slash = strrchrW(new_path,'/')))
Aric Stewart1d5adff2005-12-03 18:02:15 +0100275 {
276 ret = FALSE;
277 break;
278 }
279
280 len = slash - new_path;
281 new_path[len] = 0;
282 if (!create_full_path(new_path))
283 {
284 ret = FALSE;
285 break;
286 }
287 new_path[len] = '\\';
288 }
289 HeapFree(GetProcessHeap(),0,new_path);
290 return ret;
291}
292
Eric Pouechb2f079b2003-02-27 01:41:21 +0000293void WCMD_create_dir (void) {
Dave Pickles74f440e1999-06-06 15:24:04 +0000294
Markus Amsler765ff5d2006-10-31 21:11:28 +0100295 if (param1[0] == 0x00) {
Jason Edmeades5cc492c2007-06-03 22:07:39 +0100296 WCMD_output (WCMD_LoadMessage(WCMD_NOARG));
Markus Amsler765ff5d2006-10-31 21:11:28 +0100297 return;
298 }
Aric Stewart1d5adff2005-12-03 18:02:15 +0100299 if (!create_full_path(param1)) WCMD_print_error ();
Dave Pickles74f440e1999-06-06 15:24:04 +0000300}
301
302/****************************************************************************
303 * WCMD_delete
304 *
305 * Delete a file or wildcarded set.
306 *
Jason Edmeadesb822e732007-02-27 23:20:55 +0000307 * Note on /A:
308 * - Testing shows /A is repeatable, eg. /a-r /ar matches all files
309 * - Each set is a pattern, eg /ahr /as-r means
310 * readonly+hidden OR nonreadonly system files
311 * - The '-' applies to a single field, ie /a:-hr means read only
312 * non-hidden files
Dave Pickles74f440e1999-06-06 15:24:04 +0000313 */
314
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100315BOOL WCMD_delete (WCHAR *command, BOOL expectDir) {
Dave Pickles74f440e1999-06-06 15:24:04 +0000316
Jason Edmeadesa97d1202007-03-04 22:35:09 +0000317 int argno = 0;
318 int argsProcessed = 0;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100319 WCHAR *argN = command;
Jason Edmeades68b11d12007-04-23 23:24:38 +0100320 BOOL foundAny = FALSE;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100321 static const WCHAR parmA[] = {'/','A','\0'};
322 static const WCHAR parmQ[] = {'/','Q','\0'};
323 static const WCHAR parmP[] = {'/','P','\0'};
324 static const WCHAR parmS[] = {'/','S','\0'};
325 static const WCHAR parmF[] = {'/','F','\0'};
Jason Edmeades68b11d12007-04-23 23:24:38 +0100326
327 /* If not recursing, clear error flag */
328 if (expectDir) errorlevel = 0;
Dave Pickles74f440e1999-06-06 15:24:04 +0000329
Jason Edmeadesa97d1202007-03-04 22:35:09 +0000330 /* Loop through all args */
331 while (argN) {
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100332 WCHAR *thisArg = WCMD_parameter (command, argno++, &argN);
333 WCHAR argCopy[MAX_PATH];
Jason Edmeades68b11d12007-04-23 23:24:38 +0100334
Jason Edmeadesa97d1202007-03-04 22:35:09 +0000335 if (argN && argN[0] != '/') {
Jason Edmeades409368e2007-02-27 23:19:24 +0000336
Jason Edmeadesa97d1202007-03-04 22:35:09 +0000337 WIN32_FIND_DATA fd;
338 HANDLE hff;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100339 WCHAR fpath[MAX_PATH];
340 WCHAR *p;
Jason Edmeades68b11d12007-04-23 23:24:38 +0100341 BOOL handleParm = TRUE;
342 BOOL found = FALSE;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100343 static const WCHAR anyExt[]= {'.','*','\0'};
Jason Edmeades409368e2007-02-27 23:19:24 +0000344
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100345 strcpyW(argCopy, thisArg);
346 WINE_TRACE("del: Processing arg %s (quals:%s)\n",
347 wine_dbgstr_w(argCopy), wine_dbgstr_w(quals));
Jason Edmeadesa97d1202007-03-04 22:35:09 +0000348 argsProcessed++;
Jason Edmeades409368e2007-02-27 23:19:24 +0000349
Jason Edmeadesa97d1202007-03-04 22:35:09 +0000350 /* If filename part of parameter is * or *.*, prompt unless
351 /Q supplied. */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100352 if ((strstrW (quals, parmQ) == NULL) && (strstrW (quals, parmP) == NULL)) {
Jason Edmeades409368e2007-02-27 23:19:24 +0000353
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100354 WCHAR drive[10];
355 WCHAR dir[MAX_PATH];
356 WCHAR fname[MAX_PATH];
357 WCHAR ext[MAX_PATH];
Jason Edmeades409368e2007-02-27 23:19:24 +0000358
Jason Edmeadesa97d1202007-03-04 22:35:09 +0000359 /* Convert path into actual directory spec */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100360 GetFullPathName (argCopy, sizeof(fpath)/sizeof(WCHAR), fpath, NULL);
Jason Edmeadesa97d1202007-03-04 22:35:09 +0000361 WCMD_splitpath(fpath, drive, dir, fname, ext);
Jason Edmeades409368e2007-02-27 23:19:24 +0000362
Jason Edmeadesa97d1202007-03-04 22:35:09 +0000363 /* Only prompt for * and *.*, not *a, a*, *.a* etc */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100364 if ((strcmpW(fname, starW) == 0) &&
365 (*ext == 0x00 || (strcmpW(ext, anyExt) == 0))) {
Jason Edmeadesa97d1202007-03-04 22:35:09 +0000366 BOOL ok;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100367 WCHAR question[MAXSTRING];
368 static const WCHAR fmt[] = {'%','s',' ','\0'};
Jason Edmeades10c3764d2007-02-27 23:20:25 +0000369
Jason Edmeades68b11d12007-04-23 23:24:38 +0100370 /* Note: Flag as found, to avoid file not found message */
371 found = TRUE;
372
Jason Edmeadesa97d1202007-03-04 22:35:09 +0000373 /* Ask for confirmation */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100374 wsprintf(question, fmt, fpath);
Jason Edmeades5cc492c2007-06-03 22:07:39 +0100375 ok = WCMD_ask_confirm(question, TRUE, NULL);
Jason Edmeades10c3764d2007-02-27 23:20:25 +0000376
Jason Edmeadesa97d1202007-03-04 22:35:09 +0000377 /* Abort if answer is 'N' */
378 if (!ok) continue;
379 }
380 }
Jason Edmeadesb822e732007-02-27 23:20:55 +0000381
Jason Edmeades68b11d12007-04-23 23:24:38 +0100382 /* First, try to delete in the current directory */
383 hff = FindFirstFile (argCopy, &fd);
Jason Edmeadesa97d1202007-03-04 22:35:09 +0000384 if (hff == INVALID_HANDLE_VALUE) {
Jason Edmeades68b11d12007-04-23 23:24:38 +0100385 handleParm = FALSE;
386 } else {
387 found = TRUE;
Jason Edmeadesa97d1202007-03-04 22:35:09 +0000388 }
Jason Edmeades68b11d12007-04-23 23:24:38 +0100389
Jason Edmeadesa97d1202007-03-04 22:35:09 +0000390 /* Support del <dirname> by just deleting all files dirname\* */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100391 if (handleParm && (strchrW(argCopy,'*') == NULL) && (strchrW(argCopy,'?') == NULL)
Jason Edmeadesa97d1202007-03-04 22:35:09 +0000392 && (fd.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY)) {
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100393 WCHAR modifiedParm[MAX_PATH];
394 static const WCHAR slashStar[] = {'\\','*','\0'};
395
396 strcpyW(modifiedParm, argCopy);
397 strcatW(modifiedParm, slashStar);
Jason Edmeadesa97d1202007-03-04 22:35:09 +0000398 FindClose(hff);
Jason Edmeades68b11d12007-04-23 23:24:38 +0100399 found = TRUE;
400 WCMD_delete(modifiedParm, FALSE);
Jason Edmeadesb822e732007-02-27 23:20:55 +0000401
Jason Edmeades68b11d12007-04-23 23:24:38 +0100402 } else if (handleParm) {
Jason Edmeadesb822e732007-02-27 23:20:55 +0000403
Jason Edmeadesa97d1202007-03-04 22:35:09 +0000404 /* Build the filename to delete as <supplied directory>\<findfirst filename> */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100405 strcpyW (fpath, argCopy);
Jason Edmeadesa97d1202007-03-04 22:35:09 +0000406 do {
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100407 p = strrchrW (fpath, '\\');
Jason Edmeadesa97d1202007-03-04 22:35:09 +0000408 if (p != NULL) {
409 *++p = '\0';
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100410 strcatW (fpath, fd.cFileName);
Jason Edmeadesb822e732007-02-27 23:20:55 +0000411 }
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100412 else strcpyW (fpath, fd.cFileName);
Jason Edmeadesa97d1202007-03-04 22:35:09 +0000413 if (!(fd.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY)) {
414 BOOL ok = TRUE;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100415 WCHAR *nextA = strstrW (quals, parmA);
Jason Edmeadesb822e732007-02-27 23:20:55 +0000416
Jason Edmeadesa97d1202007-03-04 22:35:09 +0000417 /* Handle attribute matching (/A) */
418 if (nextA != NULL) {
419 ok = FALSE;
420 while (nextA != NULL && !ok) {
Jason Edmeadesb822e732007-02-27 23:20:55 +0000421
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100422 WCHAR *thisA = (nextA+2);
Jason Edmeadesa97d1202007-03-04 22:35:09 +0000423 BOOL stillOK = TRUE;
424
425 /* Skip optional : */
426 if (*thisA == ':') thisA++;
427
428 /* Parse each of the /A[:]xxx in turn */
429 while (*thisA && *thisA != '/') {
430 BOOL negate = FALSE;
431 BOOL attribute = FALSE;
432
433 /* Match negation of attribute first */
434 if (*thisA == '-') {
435 negate=TRUE;
436 thisA++;
437 }
438
439 /* Match attribute */
440 switch (*thisA) {
441 case 'R': attribute = (fd.dwFileAttributes & FILE_ATTRIBUTE_READONLY);
442 break;
443 case 'H': attribute = (fd.dwFileAttributes & FILE_ATTRIBUTE_HIDDEN);
444 break;
445 case 'S': attribute = (fd.dwFileAttributes & FILE_ATTRIBUTE_SYSTEM);
446 break;
447 case 'A': attribute = (fd.dwFileAttributes & FILE_ATTRIBUTE_ARCHIVE);
448 break;
449 default:
Jason Edmeades5cc492c2007-06-03 22:07:39 +0100450 WCMD_output (WCMD_LoadMessage(WCMD_SYNTAXERR));
Jason Edmeadesa97d1202007-03-04 22:35:09 +0000451 }
452
453 /* Now check result, keeping a running boolean about whether it
454 matches all parsed attribues so far */
455 if (attribute && !negate) {
456 stillOK = stillOK;
457 } else if (!attribute && negate) {
458 stillOK = stillOK;
459 } else {
460 stillOK = FALSE;
461 }
462 thisA++;
463 }
464
465 /* Save the running total as the final result */
466 ok = stillOK;
467
468 /* Step on to next /A set */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100469 nextA = strstrW (nextA+1, parmA);
Jason Edmeadesa97d1202007-03-04 22:35:09 +0000470 }
471 }
472
473 /* /P means prompt for each file */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100474 if (ok && strstrW (quals, parmP) != NULL) {
475 WCHAR question[MAXSTRING];
Jason Edmeadesa97d1202007-03-04 22:35:09 +0000476
477 /* Ask for confirmation */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100478 wsprintf(question, WCMD_LoadMessage(WCMD_DELPROMPT), fpath);
Jason Edmeades5cc492c2007-06-03 22:07:39 +0100479 ok = WCMD_ask_confirm(question, FALSE, NULL);
Jason Edmeadesa97d1202007-03-04 22:35:09 +0000480 }
481
482 /* Only proceed if ok to */
483 if (ok) {
484
485 /* If file is read only, and /F supplied, delete it */
486 if (fd.dwFileAttributes & FILE_ATTRIBUTE_READONLY &&
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100487 strstrW (quals, parmF) != NULL) {
Jason Edmeadesa97d1202007-03-04 22:35:09 +0000488 SetFileAttributes(fpath, fd.dwFileAttributes & ~FILE_ATTRIBUTE_READONLY);
489 }
490
491 /* Now do the delete */
492 if (!DeleteFile (fpath)) WCMD_print_error ();
493 }
494
495 }
496 } while (FindNextFile(hff, &fd) != 0);
497 FindClose (hff);
Jason Edmeadesb822e732007-02-27 23:20:55 +0000498 }
Jason Edmeades68b11d12007-04-23 23:24:38 +0100499
Francois Gouget5a8fc342007-04-30 02:06:11 +0200500 /* Now recurse into all subdirectories handling the parameter in the same way */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100501 if (strstrW (quals, parmS) != NULL) {
Jason Edmeades68b11d12007-04-23 23:24:38 +0100502
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100503 WCHAR thisDir[MAX_PATH];
Jason Edmeades68b11d12007-04-23 23:24:38 +0100504 int cPos;
505
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100506 WCHAR drive[10];
507 WCHAR dir[MAX_PATH];
508 WCHAR fname[MAX_PATH];
509 WCHAR ext[MAX_PATH];
Jason Edmeades68b11d12007-04-23 23:24:38 +0100510
511 /* Convert path into actual directory spec */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100512 GetFullPathName (argCopy, sizeof(thisDir)/sizeof(WCHAR), thisDir, NULL);
Jason Edmeades68b11d12007-04-23 23:24:38 +0100513 WCMD_splitpath(thisDir, drive, dir, fname, ext);
514
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100515 strcpyW(thisDir, drive);
516 strcatW(thisDir, dir);
517 cPos = strlenW(thisDir);
Jason Edmeades68b11d12007-04-23 23:24:38 +0100518
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100519 WINE_TRACE("Searching recursively in '%s'\n", wine_dbgstr_w(thisDir));
Jason Edmeades68b11d12007-04-23 23:24:38 +0100520
521 /* Append '*' to the directory */
522 thisDir[cPos] = '*';
523 thisDir[cPos+1] = 0x00;
524
525 hff = FindFirstFile (thisDir, &fd);
526
527 /* Remove residual '*' */
528 thisDir[cPos] = 0x00;
529
530 if (hff != INVALID_HANDLE_VALUE) {
531 DIRECTORY_STACK *allDirs = NULL;
532 DIRECTORY_STACK *lastEntry = NULL;
533
534 do {
535 if ((fd.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) &&
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100536 (strcmpW(fd.cFileName, dotdotW) != 0) &&
537 (strcmpW(fd.cFileName, dotW) != 0)) {
Jason Edmeades68b11d12007-04-23 23:24:38 +0100538
539 DIRECTORY_STACK *nextDir;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100540 WCHAR subParm[MAX_PATH];
Jason Edmeades68b11d12007-04-23 23:24:38 +0100541
542 /* Work out search parameter in sub dir */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100543 strcpyW (subParm, thisDir);
544 strcatW (subParm, fd.cFileName);
545 strcatW (subParm, slashW);
546 strcatW (subParm, fname);
547 strcatW (subParm, ext);
548 WINE_TRACE("Recursive, Adding to search list '%s'\n", wine_dbgstr_w(subParm));
Jason Edmeades68b11d12007-04-23 23:24:38 +0100549
550 /* Allocate memory, add to list */
Michael Stefaniuc6519a832007-06-27 00:12:22 +0200551 nextDir = HeapAlloc(GetProcessHeap(),0,sizeof(DIRECTORY_STACK));
Jason Edmeades68b11d12007-04-23 23:24:38 +0100552 if (allDirs == NULL) allDirs = nextDir;
553 if (lastEntry != NULL) lastEntry->next = nextDir;
554 lastEntry = nextDir;
555 nextDir->next = NULL;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100556 nextDir->dirName = HeapAlloc(GetProcessHeap(),0,
557 (strlenW(subParm)+1) * sizeof(WCHAR));
558 strcpyW(nextDir->dirName, subParm);
Jason Edmeades68b11d12007-04-23 23:24:38 +0100559 }
560 } while (FindNextFile(hff, &fd) != 0);
561 FindClose (hff);
562
563 /* Go through each subdir doing the delete */
564 while (allDirs != NULL) {
565 DIRECTORY_STACK *tempDir;
566
567 tempDir = allDirs->next;
568 found |= WCMD_delete (allDirs->dirName, FALSE);
569
570 HeapFree(GetProcessHeap(),0,allDirs->dirName);
571 HeapFree(GetProcessHeap(),0,allDirs);
572 allDirs = tempDir;
573 }
574 }
575 }
576 /* Keep running total to see if any found, and if not recursing
577 issue error message */
578 if (expectDir) {
579 if (!found) {
580 errorlevel = 1;
Jason Edmeades5cc492c2007-06-03 22:07:39 +0100581 WCMD_output (WCMD_LoadMessage(WCMD_FILENOTFOUND), argCopy);
Jason Edmeades68b11d12007-04-23 23:24:38 +0100582 }
583 }
584 foundAny |= found;
Dave Pickles74f440e1999-06-06 15:24:04 +0000585 }
Jason Edmeadesa97d1202007-03-04 22:35:09 +0000586 }
587
588 /* Handle no valid args */
589 if (argsProcessed == 0) {
Jason Edmeades5cc492c2007-06-03 22:07:39 +0100590 WCMD_output (WCMD_LoadMessage(WCMD_NOARG));
Jason Edmeadesa97d1202007-03-04 22:35:09 +0000591 }
Jason Edmeades68b11d12007-04-23 23:24:38 +0100592
593 return foundAny;
Dave Pickles74f440e1999-06-06 15:24:04 +0000594}
595
596/****************************************************************************
597 * WCMD_echo
598 *
599 * Echo input to the screen (or not). We don't try to emulate the bugs
600 * in DOS (try typing "ECHO ON AGAIN" for an example).
601 */
602
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100603void WCMD_echo (const WCHAR *command) {
Dave Pickles74f440e1999-06-06 15:24:04 +0000604
Alexandre Julliardabfe1052007-03-01 12:43:19 +0100605 int count;
Dave Pickles74f440e1999-06-06 15:24:04 +0000606
Mike McCormack8d020102004-03-18 04:01:32 +0000607 if ((command[0] == '.') && (command[1] == 0)) {
608 WCMD_output (newline);
609 return;
610 }
611 if (command[0]==' ')
612 command++;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100613 count = strlenW(command);
Dave Pickles74f440e1999-06-06 15:24:04 +0000614 if (count == 0) {
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100615 if (echo_mode) WCMD_output (WCMD_LoadMessage(WCMD_ECHOPROMPT), onW);
616 else WCMD_output (WCMD_LoadMessage(WCMD_ECHOPROMPT), offW);
Dave Pickles74f440e1999-06-06 15:24:04 +0000617 return;
618 }
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100619 if (lstrcmpiW(command, onW) == 0) {
Dave Pickles74f440e1999-06-06 15:24:04 +0000620 echo_mode = 1;
621 return;
622 }
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100623 if (lstrcmpiW(command, offW) == 0) {
Dave Pickles74f440e1999-06-06 15:24:04 +0000624 echo_mode = 0;
625 return;
626 }
Vincent Béron9a624912002-05-31 23:06:46 +0000627 WCMD_output_asis (command);
Dave Pickles74f440e1999-06-06 15:24:04 +0000628 WCMD_output (newline);
629
630}
631
Dave Pickles036a9f71999-07-10 11:36:45 +0000632/**************************************************************************
Dave Pickles74f440e1999-06-06 15:24:04 +0000633 * WCMD_for
634 *
635 * Batch file loop processing.
Jason Edmeades54d890c2007-06-15 20:59:28 +0100636 *
637 * On entry: cmdList contains the syntax up to the set
638 * next cmdList and all in that bracket contain the set data
639 * next cmdlist contains the DO cmd
640 * following that is either brackets or && entries (as per if)
641 *
Dave Pickles036a9f71999-07-10 11:36:45 +0000642 * FIXME: We don't exhaustively check syntax. Any command which works in MessDOS
643 * will probably work here, but the reverse is not necessarily the case...
Dave Pickles74f440e1999-06-06 15:24:04 +0000644 */
645
Jason Edmeades8f12d8b2007-06-15 20:59:21 +0100646void WCMD_for (WCHAR *p, CMD_LIST **cmdList) {
Dave Pickles74f440e1999-06-06 15:24:04 +0000647
Alexandre Julliardabfe1052007-03-01 12:43:19 +0100648 WIN32_FIND_DATA fd;
649 HANDLE hff;
Alexandre Julliardabfe1052007-03-01 12:43:19 +0100650 int i;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100651 const WCHAR inW[] = {'i', 'n', '\0'};
Jason Edmeades54d890c2007-06-15 20:59:28 +0100652 const WCHAR doW[] = {'d', 'o', ' ','\0'};
653 CMD_LIST *setStart, *thisSet, *cmdStart, *cmdEnd;
654 WCHAR variable[4];
655 WCHAR *firstCmd;
656 int thisDepth;
Jason Edmeades196fb102007-06-15 20:59:29 +0100657 BOOL isDirs = FALSE;
Dave Pickles036a9f71999-07-10 11:36:45 +0000658
Jason Edmeades54d890c2007-06-15 20:59:28 +0100659 /* Check:
660 the first line includes the % variable name as first parm
661 we have been provided with more parts to the command
662 and there is at least some set data
663 and IN as the one after that */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100664 if (lstrcmpiW (WCMD_parameter (p, 1, NULL), inW)
Jason Edmeades54d890c2007-06-15 20:59:28 +0100665 || (*cmdList) == NULL
666 || (*cmdList)->nextcommand == NULL
667 || (param1[0] != '%')
668 || (strlenW(param1) > 3)) {
Jason Edmeades5cc492c2007-06-03 22:07:39 +0100669 WCMD_output (WCMD_LoadMessage(WCMD_SYNTAXERR));
Dave Pickles5f8f4f71999-06-26 10:24:08 +0000670 return;
671 }
Dave Pickles74f440e1999-06-06 15:24:04 +0000672
Jason Edmeades54d890c2007-06-15 20:59:28 +0100673 /* Save away where the set of data starts and the variable */
674 strcpyW(variable, param1);
675 thisDepth = (*cmdList)->bracketDepth;
676 *cmdList = (*cmdList)->nextcommand;
677 setStart = (*cmdList);
Dave Pickles036a9f71999-07-10 11:36:45 +0000678
Jason Edmeades54d890c2007-06-15 20:59:28 +0100679 /* Skip until the close bracket */
680 WINE_TRACE("Searching %p as the set\n", *cmdList);
681 while (*cmdList &&
682 (*cmdList)->command != NULL &&
683 (*cmdList)->bracketDepth > thisDepth) {
684 WINE_TRACE("Skipping %p which is part of the set\n", *cmdList);
685 *cmdList = (*cmdList)->nextcommand;
Dave Pickles036a9f71999-07-10 11:36:45 +0000686 }
Jason Edmeades54d890c2007-06-15 20:59:28 +0100687
688 /* Skip the close bracket, if there is one */
689 if (*cmdList) *cmdList = (*cmdList)->nextcommand;
690
691 /* Syntax error if missing close bracket, or nothing following it
692 and once we have the complete set, we expect a DO */
693 WINE_TRACE("Looking for 'do' in %p\n", *cmdList);
694 if ((*cmdList == NULL) ||
695 (CompareString (LOCALE_USER_DEFAULT, NORM_IGNORECASE | SORT_STRINGSORT,
696 (*cmdList)->command, 3, doW, -1) != 2)) {
697 WCMD_output (WCMD_LoadMessage(WCMD_SYNTAXERR));
698 return;
699 }
700
701 /* Save away the starting position for the commands (and offset for the
702 first one */
703 cmdStart = *cmdList;
704 cmdEnd = *cmdList;
705 firstCmd = (*cmdList)->command + 3; /* Skip 'do ' */
706
707 thisSet = setStart;
708 /* Loop through all set entries */
709 while (thisSet &&
710 thisSet->command != NULL &&
711 thisSet->bracketDepth >= thisDepth) {
712
713 /* Loop through all entries on the same line */
714 WCHAR *item;
715
716 WINE_TRACE("Processing for set %p\n", thisSet);
717 i = 0;
718 while (*(item = WCMD_parameter (thisSet->command, i, NULL))) {
719
720 /*
721 * If the parameter within the set has a wildcard then search for matching files
722 * otherwise do a literal substitution.
723 */
724 static const WCHAR wildcards[] = {'*','?','\0'};
725 CMD_LIST *thisCmdStart = cmdStart;
726
727 WINE_TRACE("Processing for item '%s'\n", wine_dbgstr_w(item));
728 if (strpbrkW (item, wildcards)) {
729 hff = FindFirstFile (item, &fd);
730 if (hff != INVALID_HANDLE_VALUE) {
731 do {
Jason Edmeades196fb102007-06-15 20:59:29 +0100732 BOOL isDirectory = (fd.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY);
733 if ((isDirs && isDirectory) ||
734 (!isDirs && !isDirectory))
735 {
736 thisCmdStart = cmdStart;
737 WINE_TRACE("Processing FOR filename %s\n", wine_dbgstr_w(fd.cFileName));
738 WCMD_part_execute (&thisCmdStart, firstCmd, variable,
739 fd.cFileName, FALSE, TRUE);
740 }
Jason Edmeades54d890c2007-06-15 20:59:28 +0100741
742 } while (FindNextFile(hff, &fd) != 0);
743 FindClose (hff);
744 }
745 } else {
746 WCMD_part_execute(&thisCmdStart, firstCmd, variable, item, FALSE, TRUE);
747 }
748
749 WINE_TRACE("Post-command, cmdEnd = %p\n", cmdEnd);
750 cmdEnd = thisCmdStart;
751 i++;
752 }
753
754 /* Move onto the next set line */
755 thisSet = thisSet->nextcommand;
756 }
757
758 /* When the loop ends, either something like a GOTO or EXIT /b has terminated
759 all processing, OR it should be pointing to the end of && processing OR
760 it should be pointing at the NULL end of bracket for the DO. The return
761 value needs to be the NEXT command to execute, which it either is, or
762 we need to step over the closing bracket */
763 *cmdList = cmdEnd;
764 if (cmdEnd && cmdEnd->command == NULL) *cmdList = cmdEnd->nextcommand;
765}
766
767
768/*****************************************************************************
769 * WCMD_part_execute
770 *
771 * Execute a command, and any && or bracketed follow on to the command. The
772 * first command to be executed may not be at the front of the
Jason Edmeades8e089832007-07-27 21:43:38 +0100773 * commands->thiscommand string (eg. it may point after a DO or ELSE)
Jason Edmeades54d890c2007-06-15 20:59:28 +0100774 */
775void WCMD_part_execute(CMD_LIST **cmdList, WCHAR *firstcmd, WCHAR *variable,
776 WCHAR *value, BOOL isIF, BOOL conditionTRUE) {
777
778 CMD_LIST *curPosition = *cmdList;
779 int myDepth = (*cmdList)->bracketDepth;
780
781 WINE_TRACE("cmdList(%p), firstCmd(%p), with '%s'='%s', doIt(%d)\n",
782 cmdList, wine_dbgstr_w(firstcmd),
783 wine_dbgstr_w(variable), wine_dbgstr_w(value),
784 conditionTRUE);
785
786 /* Skip leading whitespace between condition and the command */
787 while (firstcmd && *firstcmd && (*firstcmd==' ' || *firstcmd=='\t')) firstcmd++;
788
789 /* Process the first command, if there is one */
790 if (conditionTRUE && firstcmd && *firstcmd) {
791 WCHAR *command = WCMD_strdupW(firstcmd);
792 WCMD_execute (firstcmd, variable, value, cmdList);
793 free (command);
794 }
795
796
Francois Gougeta3317a52007-07-05 16:10:30 +0200797 /* If it didn't move the position, step to next command */
Jason Edmeades54d890c2007-06-15 20:59:28 +0100798 if (curPosition == *cmdList) *cmdList = (*cmdList)->nextcommand;
799
800 /* Process any other parts of the command */
801 if (*cmdList) {
802 BOOL processThese = TRUE;
803
804 if (isIF) processThese = conditionTRUE;
805
806 while (*cmdList) {
807 const WCHAR ifElse[] = {'e','l','s','e',' ','\0'};
808
809 /* execute all appropriate commands */
810 curPosition = *cmdList;
811
812 WINE_TRACE("Processing cmdList(%p) - &(%d) bd(%d / %d)\n",
813 *cmdList,
814 (*cmdList)->isAmphersand,
815 (*cmdList)->bracketDepth, myDepth);
816
817 /* Execute any appended to the statement with &&'s */
818 if ((*cmdList)->isAmphersand) {
819 if (processThese) {
820 WCMD_execute ((*cmdList)->command, variable, value, cmdList);
821 }
822 if (curPosition == *cmdList) *cmdList = (*cmdList)->nextcommand;
823
824 /* Execute any appended to the statement with (...) */
825 } else if ((*cmdList)->bracketDepth > myDepth) {
826 if (processThese) {
827 *cmdList = WCMD_process_commands(*cmdList, TRUE, variable, value);
828 WINE_TRACE("Back from processing commands, (next = %p)\n", *cmdList);
829 }
830 if (curPosition == *cmdList) *cmdList = (*cmdList)->nextcommand;
831
832 /* End of the command - does 'ELSE ' follow as the next command? */
833 } else {
834 if (isIF && CompareString (LOCALE_USER_DEFAULT,
835 NORM_IGNORECASE | SORT_STRINGSORT,
836 (*cmdList)->command, 5, ifElse, -1) == 2) {
837
838 /* Swap between if and else processing */
839 processThese = !processThese;
840
841 /* Process the ELSE part */
842 if (processThese) {
843 WCHAR *cmd = ((*cmdList)->command) + strlenW(ifElse);
844
845 /* Skip leading whitespace between condition and the command */
846 while (*cmd && (*cmd==' ' || *cmd=='\t')) cmd++;
847 if (*cmd) {
848 WCMD_execute (cmd, variable, value, cmdList);
849 }
850 }
851 if (curPosition == *cmdList) *cmdList = (*cmdList)->nextcommand;
852 } else {
853 WINE_TRACE("Found end of this IF statement (next = %p)\n", *cmdList);
854 break;
855 }
856 }
857 }
858 }
859 return;
Dave Pickles036a9f71999-07-10 11:36:45 +0000860}
861
Dave Picklesebecf502000-08-01 22:02:18 +0000862/*****************************************************************************
863 * WCMD_Execute
864 *
Dave Pickles036a9f71999-07-10 11:36:45 +0000865 * Execute a command after substituting variable text for the supplied parameter
866 */
867
Jason Edmeades8f12d8b2007-06-15 20:59:21 +0100868void WCMD_execute (WCHAR *orig_cmd, WCHAR *param, WCHAR *subst, CMD_LIST **cmdList) {
Dave Pickles036a9f71999-07-10 11:36:45 +0000869
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100870 WCHAR *new_cmd, *p, *s, *dup;
Alexandre Julliardabfe1052007-03-01 12:43:19 +0100871 int size;
Dave Pickles036a9f71999-07-10 11:36:45 +0000872
Jason Edmeades54d890c2007-06-15 20:59:28 +0100873 if (param) {
874 size = (strlenW (orig_cmd) + 1) * sizeof(WCHAR);
875 new_cmd = (WCHAR *) LocalAlloc (LMEM_FIXED | LMEM_ZEROINIT, size);
876 dup = s = WCMD_strdupW(orig_cmd);
Dave Pickles036a9f71999-07-10 11:36:45 +0000877
Jason Edmeades54d890c2007-06-15 20:59:28 +0100878 while ((p = strstrW (s, param))) {
879 *p = '\0';
880 size += strlenW (subst) * sizeof(WCHAR);
881 new_cmd = (WCHAR *) LocalReAlloc ((HANDLE)new_cmd, size, 0);
882 strcatW (new_cmd, s);
883 strcatW (new_cmd, subst);
884 s = p + strlenW (param);
885 }
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100886 strcatW (new_cmd, s);
Jason Edmeades54d890c2007-06-15 20:59:28 +0100887 WCMD_process_command (new_cmd, cmdList);
888 free (dup);
889 LocalFree ((HANDLE)new_cmd);
890 } else {
891 WCMD_process_command (orig_cmd, cmdList);
Dave Pickles036a9f71999-07-10 11:36:45 +0000892 }
Dave Pickles036a9f71999-07-10 11:36:45 +0000893}
894
Dave Pickles74f440e1999-06-06 15:24:04 +0000895
896/**************************************************************************
897 * WCMD_give_help
898 *
899 * Simple on-line help. Help text is stored in the resource file.
900 */
901
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100902void WCMD_give_help (WCHAR *command) {
Dave Pickles74f440e1999-06-06 15:24:04 +0000903
Alexandre Julliardabfe1052007-03-01 12:43:19 +0100904 int i;
Dave Pickles74f440e1999-06-06 15:24:04 +0000905
906 command = WCMD_strtrim_leading_spaces(command);
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100907 if (strlenW(command) == 0) {
Jason Edmeadesafe4d802007-06-03 22:07:43 +0100908 WCMD_output_asis (WCMD_LoadMessage(WCMD_ALLHELP));
Dave Pickles74f440e1999-06-06 15:24:04 +0000909 }
910 else {
911 for (i=0; i<=WCMD_EXIT; i++) {
912 if (CompareString (LOCALE_USER_DEFAULT, NORM_IGNORECASE | SORT_STRINGSORT,
Dave Picklesebecf502000-08-01 22:02:18 +0000913 param1, -1, inbuilt[i], -1) == 2) {
Jason Edmeadesafe4d802007-06-03 22:07:43 +0100914 WCMD_output_asis (WCMD_LoadMessage(i));
Dave Picklesebecf502000-08-01 22:02:18 +0000915 return;
Dave Pickles74f440e1999-06-06 15:24:04 +0000916 }
917 }
Jason Edmeades5cc492c2007-06-03 22:07:39 +0100918 WCMD_output (WCMD_LoadMessage(WCMD_NOCMDHELP), param1);
Dave Pickles74f440e1999-06-06 15:24:04 +0000919 }
920 return;
921}
922
923/****************************************************************************
Dave Pickles5f8f4f71999-06-26 10:24:08 +0000924 * WCMD_go_to
925 *
926 * Batch file jump instruction. Not the most efficient algorithm ;-)
927 * Prints error message if the specified label cannot be found - the file pointer is
928 * then at EOF, effectively stopping the batch file.
929 * FIXME: DOS is supposed to allow labels with spaces - we don't.
930 */
931
Jason Edmeades929a92f2007-06-15 20:59:22 +0100932void WCMD_goto (CMD_LIST **cmdList) {
Dave Pickles5f8f4f71999-06-26 10:24:08 +0000933
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100934 WCHAR string[MAX_PATH];
Dave Pickles5f8f4f71999-06-26 10:24:08 +0000935
Jason Edmeades929a92f2007-06-15 20:59:22 +0100936 /* Do not process any more parts of a processed multipart or multilines command */
937 *cmdList = NULL;
938
Markus Amsler765ff5d2006-10-31 21:11:28 +0100939 if (param1[0] == 0x00) {
Jason Edmeades5cc492c2007-06-03 22:07:39 +0100940 WCMD_output (WCMD_LoadMessage(WCMD_NOARG));
Markus Amsler765ff5d2006-10-31 21:11:28 +0100941 return;
942 }
Dave Pickles5f8f4f71999-06-26 10:24:08 +0000943 if (context != NULL) {
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100944 WCHAR *paramStart = param1;
945 static const WCHAR eofW[] = {':','e','o','f','\0'};
Jason Edmeades54829242007-02-20 18:00:37 +0000946
947 /* Handle special :EOF label */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100948 if (lstrcmpiW (eofW, param1) == 0) {
Jason Edmeades54829242007-02-20 18:00:37 +0000949 context -> skip_rest = TRUE;
950 return;
951 }
952
Jason Edmeades327d7ef2007-02-23 22:19:25 +0000953 /* Support goto :label as well as goto label */
954 if (*paramStart == ':') paramStart++;
955
Dave Pickles5f8f4f71999-06-26 10:24:08 +0000956 SetFilePointer (context -> h, 0, NULL, FILE_BEGIN);
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100957 while (WCMD_fgets (string, sizeof(string)/sizeof(WCHAR), context -> h)) {
958 if ((string[0] == ':') && (lstrcmpiW (&string[1], paramStart) == 0)) return;
Dave Pickles5f8f4f71999-06-26 10:24:08 +0000959 }
Jason Edmeades5cc492c2007-06-03 22:07:39 +0100960 WCMD_output (WCMD_LoadMessage(WCMD_NOTARGET));
Dave Pickles5f8f4f71999-06-26 10:24:08 +0000961 }
962 return;
963}
964
Jason Edmeades365f86f2007-02-23 22:17:28 +0000965/*****************************************************************************
966 * WCMD_pushd
967 *
968 * Push a directory onto the stack
969 */
970
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100971void WCMD_pushd (WCHAR *command) {
Jason Edmeades365f86f2007-02-23 22:17:28 +0000972 struct env_stack *curdir;
Jason Edmeades365f86f2007-02-23 22:17:28 +0000973 WCHAR *thisdir;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100974 static const WCHAR parmD[] = {'/','D','\0'};
Jason Edmeades365f86f2007-02-23 22:17:28 +0000975
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100976 if (strchrW(command, '/') != NULL) {
Jason Edmeades91aa8a02007-03-08 00:44:46 +0000977 SetLastError(ERROR_INVALID_PARAMETER);
978 WCMD_print_error();
979 return;
980 }
981
Jason Edmeades365f86f2007-02-23 22:17:28 +0000982 curdir = LocalAlloc (LMEM_FIXED, sizeof (struct env_stack));
983 thisdir = LocalAlloc (LMEM_FIXED, 1024 * sizeof(WCHAR));
984 if( !curdir || !thisdir ) {
985 LocalFree(curdir);
986 LocalFree(thisdir);
Jason Edmeades5cc492c2007-06-03 22:07:39 +0100987 WINE_ERR ("out of memory\n");
Jason Edmeades365f86f2007-02-23 22:17:28 +0000988 return;
989 }
990
Jason Edmeadesffbaede2007-03-08 00:50:14 +0000991 /* Change directory using CD code with /D parameter */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +0100992 strcpyW(quals, parmD);
Jason Edmeades365f86f2007-02-23 22:17:28 +0000993 GetCurrentDirectoryW (1024, thisdir);
Jason Edmeadese5a26bc2007-03-08 00:44:09 +0000994 errorlevel = 0;
995 WCMD_setshow_default(command);
996 if (errorlevel) {
Jason Edmeades365f86f2007-02-23 22:17:28 +0000997 LocalFree(curdir);
998 LocalFree(thisdir);
999 return;
1000 } else {
1001 curdir -> next = pushd_directories;
1002 curdir -> strings = thisdir;
Jason Edmeades8049ae12007-03-04 22:34:20 +00001003 if (pushd_directories == NULL) {
Francois Gouget540d8182007-03-12 10:35:13 +01001004 curdir -> u.stackdepth = 1;
Jason Edmeades8049ae12007-03-04 22:34:20 +00001005 } else {
Francois Gouget540d8182007-03-12 10:35:13 +01001006 curdir -> u.stackdepth = pushd_directories -> u.stackdepth + 1;
Jason Edmeades8049ae12007-03-04 22:34:20 +00001007 }
Jason Edmeades365f86f2007-02-23 22:17:28 +00001008 pushd_directories = curdir;
1009 }
1010}
1011
1012
1013/*****************************************************************************
1014 * WCMD_popd
1015 *
1016 * Pop a directory from the stack
1017 */
1018
1019void WCMD_popd (void) {
1020 struct env_stack *temp = pushd_directories;
1021
1022 if (!pushd_directories)
1023 return;
1024
1025 /* pop the old environment from the stack, and make it the current dir */
1026 pushd_directories = temp->next;
1027 SetCurrentDirectoryW(temp->strings);
1028 LocalFree (temp->strings);
1029 LocalFree (temp);
1030}
Dave Pickles5f8f4f71999-06-26 10:24:08 +00001031
1032/****************************************************************************
Dave Pickles74f440e1999-06-06 15:24:04 +00001033 * WCMD_if
1034 *
1035 * Batch file conditional.
Jason Edmeadesd2e7b402007-06-15 20:59:27 +01001036 *
1037 * On entry, cmdlist will point to command containing the IF, and optionally
1038 * the first command to execute (if brackets not found)
1039 * If &&'s were found, this may be followed by a record flagged as isAmpersand
1040 * If ('s were found, execute all within that bracket
1041 * Command may optionally be followed by an ELSE - need to skip instructions
1042 * in the else using the same logic
1043 *
Dave Pickles036a9f71999-07-10 11:36:45 +00001044 * FIXME: Much more syntax checking needed!
Dave Pickles74f440e1999-06-06 15:24:04 +00001045 */
1046
Jason Edmeades8f12d8b2007-06-15 20:59:21 +01001047void WCMD_if (WCHAR *p, CMD_LIST **cmdList) {
Dave Pickles74f440e1999-06-06 15:24:04 +00001048
Alexandre Julliardabfe1052007-03-01 12:43:19 +01001049 int negate = 0, test = 0;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001050 WCHAR condition[MAX_PATH], *command, *s;
1051 static const WCHAR notW[] = {'n','o','t','\0'};
1052 static const WCHAR errlvlW[] = {'e','r','r','o','r','l','e','v','e','l','\0'};
1053 static const WCHAR existW[] = {'e','x','i','s','t','\0'};
1054 static const WCHAR defdW[] = {'d','e','f','i','n','e','d','\0'};
1055 static const WCHAR eqeqW[] = {'=','=','\0'};
Dave Pickles74f440e1999-06-06 15:24:04 +00001056
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001057 if (!lstrcmpiW (param1, notW)) {
Dave Pickles036a9f71999-07-10 11:36:45 +00001058 negate = 1;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001059 strcpyW (condition, param2);
Alexandre Julliardabfe1052007-03-01 12:43:19 +01001060 }
Dave Pickles036a9f71999-07-10 11:36:45 +00001061 else {
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001062 strcpyW (condition, param1);
Dave Pickles036a9f71999-07-10 11:36:45 +00001063 }
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001064 if (!lstrcmpiW (condition, errlvlW)) {
1065 if (errorlevel >= atoiW(WCMD_parameter (p, 1+negate, NULL))) test = 1;
Alexandre Julliard36bf7922003-01-15 03:35:32 +00001066 WCMD_parameter (p, 2+negate, &command);
Dave Pickles036a9f71999-07-10 11:36:45 +00001067 }
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001068 else if (!lstrcmpiW (condition, existW)) {
1069 if (GetFileAttributes(WCMD_parameter (p, 1+negate, NULL)) != INVALID_FILE_ATTRIBUTES) {
Jason Edmeadesc79342a2005-02-14 11:01:35 +00001070 test = 1;
Dave Pickles036a9f71999-07-10 11:36:45 +00001071 }
Alexandre Julliard36bf7922003-01-15 03:35:32 +00001072 WCMD_parameter (p, 2+negate, &command);
Dave Pickles036a9f71999-07-10 11:36:45 +00001073 }
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001074 else if (!lstrcmpiW (condition, defdW)) {
1075 if (GetEnvironmentVariable(WCMD_parameter (p, 1+negate, NULL), NULL, 0) > 0) {
Jason Edmeades758a3972007-02-20 17:47:35 +00001076 test = 1;
1077 }
1078 WCMD_parameter (p, 2+negate, &command);
1079 }
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001080 else if ((s = strstrW (p, eqeqW))) {
Dave Pickles036a9f71999-07-10 11:36:45 +00001081 s += 2;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001082 if (!lstrcmpiW (condition, WCMD_parameter (s, 0, NULL))) test = 1;
Alexandre Julliard36bf7922003-01-15 03:35:32 +00001083 WCMD_parameter (s, 1, &command);
Dave Pickles036a9f71999-07-10 11:36:45 +00001084 }
1085 else {
Jason Edmeades5cc492c2007-06-03 22:07:39 +01001086 WCMD_output (WCMD_LoadMessage(WCMD_SYNTAXERR));
Dave Pickles036a9f71999-07-10 11:36:45 +00001087 return;
1088 }
Jason Edmeadesd2e7b402007-06-15 20:59:27 +01001089
1090 /* Process rest of IF statement which is on the same line
1091 Note: This may process all or some of the cmdList (eg a GOTO) */
Jason Edmeades54d890c2007-06-15 20:59:28 +01001092 WCMD_part_execute(cmdList, command, NULL, NULL, TRUE, (test != negate));
Dave Pickles74f440e1999-06-06 15:24:04 +00001093}
1094
1095/****************************************************************************
1096 * WCMD_move
1097 *
1098 * Move a file, directory tree or wildcarded set of files.
1099 */
1100
Eric Pouechb2f079b2003-02-27 01:41:21 +00001101void WCMD_move (void) {
Dave Pickles74f440e1999-06-06 15:24:04 +00001102
Jason Edmeadesd0a6fe12007-04-11 22:25:52 +01001103 int status;
Alexandre Julliardabfe1052007-03-01 12:43:19 +01001104 WIN32_FIND_DATA fd;
Jason Edmeadesd0a6fe12007-04-11 22:25:52 +01001105 HANDLE hff;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001106 WCHAR input[MAX_PATH];
1107 WCHAR output[MAX_PATH];
1108 WCHAR drive[10];
1109 WCHAR dir[MAX_PATH];
1110 WCHAR fname[MAX_PATH];
1111 WCHAR ext[MAX_PATH];
Dave Pickles036a9f71999-07-10 11:36:45 +00001112
Markus Amsler765ff5d2006-10-31 21:11:28 +01001113 if (param1[0] == 0x00) {
Jason Edmeades5cc492c2007-06-03 22:07:39 +01001114 WCMD_output (WCMD_LoadMessage(WCMD_NOARG));
Markus Amsler765ff5d2006-10-31 21:11:28 +01001115 return;
1116 }
1117
Jason Edmeades13c51172002-04-20 20:54:38 +00001118 /* If no destination supplied, assume current directory */
1119 if (param2[0] == 0x00) {
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001120 strcpyW(param2, dotW);
Jason Edmeades13c51172002-04-20 20:54:38 +00001121 }
1122
1123 /* If 2nd parm is directory, then use original filename */
Jason Edmeadesd0a6fe12007-04-11 22:25:52 +01001124 /* Convert partial path to full path */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001125 GetFullPathName (param1, sizeof(input)/sizeof(WCHAR), input, NULL);
1126 GetFullPathName (param2, sizeof(output)/sizeof(WCHAR), output, NULL);
1127 WINE_TRACE("Move from '%s'('%s') to '%s'\n", wine_dbgstr_w(input),
1128 wine_dbgstr_w(param1), wine_dbgstr_w(output));
Jason Edmeades13c51172002-04-20 20:54:38 +00001129
Jason Edmeadesd0a6fe12007-04-11 22:25:52 +01001130 /* Split into components */
1131 WCMD_splitpath(input, drive, dir, fname, ext);
1132
1133 hff = FindFirstFile (input, &fd);
1134 while (hff != INVALID_HANDLE_VALUE) {
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001135 WCHAR dest[MAX_PATH];
1136 WCHAR src[MAX_PATH];
Jason Edmeadesd0a6fe12007-04-11 22:25:52 +01001137 DWORD attribs;
1138
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001139 WINE_TRACE("Processing file '%s'\n", wine_dbgstr_w(fd.cFileName));
Jason Edmeadesd0a6fe12007-04-11 22:25:52 +01001140
1141 /* Build src & dest name */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001142 strcpyW(src, drive);
1143 strcatW(src, dir);
Jason Edmeadesd0a6fe12007-04-11 22:25:52 +01001144
1145 /* See if dest is an existing directory */
1146 attribs = GetFileAttributes(output);
1147 if (attribs != INVALID_FILE_ATTRIBUTES &&
1148 (attribs & FILE_ATTRIBUTE_DIRECTORY)) {
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001149 strcpyW(dest, output);
1150 strcatW(dest, slashW);
1151 strcatW(dest, fd.cFileName);
Jason Edmeadesd0a6fe12007-04-11 22:25:52 +01001152 } else {
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001153 strcpyW(dest, output);
Jason Edmeadesd0a6fe12007-04-11 22:25:52 +01001154 }
1155
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001156 strcatW(src, fd.cFileName);
Jason Edmeadesd0a6fe12007-04-11 22:25:52 +01001157
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001158 WINE_TRACE("Source '%s'\n", wine_dbgstr_w(src));
1159 WINE_TRACE("Dest '%s'\n", wine_dbgstr_w(dest));
Jason Edmeadesd0a6fe12007-04-11 22:25:52 +01001160
1161 /* Check if file is read only, otherwise move it */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001162 attribs = GetFileAttributes(src);
Jason Edmeadesd0a6fe12007-04-11 22:25:52 +01001163 if ((attribs != INVALID_FILE_ATTRIBUTES) &&
1164 (attribs & FILE_ATTRIBUTE_READONLY)) {
1165 SetLastError(ERROR_ACCESS_DENIED);
1166 status = 0;
1167 } else {
Jason Edmeades2f5bfc02007-04-11 22:25:53 +01001168 BOOL ok = TRUE;
1169
1170 /* If destination exists, prompt unless /Y supplied */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001171 if (GetFileAttributes(dest) != INVALID_FILE_ATTRIBUTES) {
Jason Edmeades2f5bfc02007-04-11 22:25:53 +01001172 BOOL force = FALSE;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001173 WCHAR copycmd[MAXSTRING];
Jason Edmeades2f5bfc02007-04-11 22:25:53 +01001174 int len;
1175
1176 /* /-Y has the highest priority, then /Y and finally the COPYCMD env. variable */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001177 if (strstrW (quals, parmNoY))
Jason Edmeades2f5bfc02007-04-11 22:25:53 +01001178 force = FALSE;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001179 else if (strstrW (quals, parmY))
Jason Edmeades2f5bfc02007-04-11 22:25:53 +01001180 force = TRUE;
1181 else {
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001182 const WCHAR copyCmdW[] = {'C','O','P','Y','C','M','D','\0'};
1183 len = GetEnvironmentVariable (copyCmdW, copycmd, sizeof(copycmd)/sizeof(WCHAR));
1184 force = (len && len < (sizeof(copycmd)/sizeof(WCHAR))
1185 && ! lstrcmpiW (copycmd, parmY));
Jason Edmeades2f5bfc02007-04-11 22:25:53 +01001186 }
1187
1188 /* Prompt if overwriting */
1189 if (!force) {
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001190 WCHAR question[MAXSTRING];
1191 WCHAR yesChar[10];
Jason Edmeades2f5bfc02007-04-11 22:25:53 +01001192
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001193 strcpyW(yesChar, WCMD_LoadMessage(WCMD_YES));
Jason Edmeades2f5bfc02007-04-11 22:25:53 +01001194
1195 /* Ask for confirmation */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001196 wsprintf(question, WCMD_LoadMessage(WCMD_OVERWRITE), dest);
Jason Edmeades5cc492c2007-06-03 22:07:39 +01001197 ok = WCMD_ask_confirm(question, FALSE, NULL);
Jason Edmeades2f5bfc02007-04-11 22:25:53 +01001198
1199 /* So delete the destination prior to the move */
1200 if (ok) {
1201 if (!DeleteFile (dest)) {
1202 WCMD_print_error ();
1203 errorlevel = 1;
1204 ok = FALSE;
1205 }
1206 }
1207 }
1208 }
1209
1210 if (ok) {
1211 status = MoveFile (src, dest);
1212 } else {
1213 status = 1; /* Anything other than 0 to prevent error msg below */
1214 }
Jason Edmeadesd0a6fe12007-04-11 22:25:52 +01001215 }
1216
1217 if (!status) {
1218 WCMD_print_error ();
1219 errorlevel = 1;
1220 }
1221
1222 /* Step on to next match */
1223 if (FindNextFile(hff, &fd) == 0) {
1224 FindClose(hff);
1225 hff = INVALID_HANDLE_VALUE;
1226 break;
1227 }
1228 }
Dave Pickles74f440e1999-06-06 15:24:04 +00001229}
1230
1231/****************************************************************************
1232 * WCMD_pause
1233 *
1234 * Wait for keyboard input.
1235 */
1236
Eric Pouechb2f079b2003-02-27 01:41:21 +00001237void WCMD_pause (void) {
Dave Pickles74f440e1999-06-06 15:24:04 +00001238
Alexandre Julliardabfe1052007-03-01 12:43:19 +01001239 DWORD count;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001240 WCHAR string[32];
Dave Pickles74f440e1999-06-06 15:24:04 +00001241
1242 WCMD_output (anykey);
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001243 WCMD_ReadFile (GetStdHandle(STD_INPUT_HANDLE), string,
1244 sizeof(string)/sizeof(WCHAR), &count, NULL);
Dave Pickles74f440e1999-06-06 15:24:04 +00001245}
1246
1247/****************************************************************************
1248 * WCMD_remove_dir
1249 *
1250 * Delete a directory.
1251 */
1252
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001253void WCMD_remove_dir (WCHAR *command) {
Dave Pickles74f440e1999-06-06 15:24:04 +00001254
Jason Edmeades0efa91d2007-03-04 22:33:27 +00001255 int argno = 0;
1256 int argsProcessed = 0;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001257 WCHAR *argN = command;
1258 static const WCHAR parmS[] = {'/','S','\0'};
1259 static const WCHAR parmQ[] = {'/','Q','\0'};
Jason Edmeades0efa91d2007-03-04 22:33:27 +00001260
1261 /* Loop through all args */
1262 while (argN) {
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001263 WCHAR *thisArg = WCMD_parameter (command, argno++, &argN);
Jason Edmeades0efa91d2007-03-04 22:33:27 +00001264 if (argN && argN[0] != '/') {
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001265 WINE_TRACE("rd: Processing arg %s (quals:%s)\n", wine_dbgstr_w(thisArg),
1266 wine_dbgstr_w(quals));
Jason Edmeades0efa91d2007-03-04 22:33:27 +00001267 argsProcessed++;
1268
1269 /* If subdirectory search not supplied, just try to remove
1270 and report error if it fails (eg if it contains a file) */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001271 if (strstrW (quals, parmS) == NULL) {
Jason Edmeades0efa91d2007-03-04 22:33:27 +00001272 if (!RemoveDirectory (thisArg)) WCMD_print_error ();
1273
1274 /* Otherwise use ShFileOp to recursively remove a directory */
1275 } else {
1276
1277 SHFILEOPSTRUCT lpDir;
1278
1279 /* Ask first */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001280 if (strstrW (quals, parmQ) == NULL) {
Jason Edmeades0efa91d2007-03-04 22:33:27 +00001281 BOOL ok;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001282 WCHAR question[MAXSTRING];
1283 static const WCHAR fmt[] = {'%','s',' ','\0'};
Jason Edmeades0efa91d2007-03-04 22:33:27 +00001284
1285 /* Ask for confirmation */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001286 wsprintf(question, fmt, thisArg);
Jason Edmeades5cc492c2007-06-03 22:07:39 +01001287 ok = WCMD_ask_confirm(question, TRUE, NULL);
Jason Edmeades0efa91d2007-03-04 22:33:27 +00001288
1289 /* Abort if answer is 'N' */
1290 if (!ok) return;
1291 }
1292
1293 /* Do the delete */
1294 lpDir.hwnd = NULL;
1295 lpDir.pTo = NULL;
1296 lpDir.pFrom = thisArg;
1297 lpDir.fFlags = FOF_SILENT | FOF_NOCONFIRMATION | FOF_NOERRORUI;
1298 lpDir.wFunc = FO_DELETE;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001299 if (SHFileOperation(&lpDir)) WCMD_print_error ();
Jason Edmeades0efa91d2007-03-04 22:33:27 +00001300 }
1301 }
1302 }
1303
1304 /* Handle no valid args */
1305 if (argsProcessed == 0) {
Jason Edmeades5cc492c2007-06-03 22:07:39 +01001306 WCMD_output (WCMD_LoadMessage(WCMD_NOARG));
Markus Amsler765ff5d2006-10-31 21:11:28 +01001307 return;
1308 }
Jason Edmeades69194ce2007-02-26 23:04:40 +00001309
Dave Pickles74f440e1999-06-06 15:24:04 +00001310}
1311
1312/****************************************************************************
1313 * WCMD_rename
1314 *
1315 * Rename a file.
Dave Pickles74f440e1999-06-06 15:24:04 +00001316 */
1317
Eric Pouechb2f079b2003-02-27 01:41:21 +00001318void WCMD_rename (void) {
Dave Pickles74f440e1999-06-06 15:24:04 +00001319
Jason Edmeades32fc3662007-04-11 22:25:51 +01001320 int status;
1321 HANDLE hff;
1322 WIN32_FIND_DATA fd;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001323 WCHAR input[MAX_PATH];
1324 WCHAR *dotDst = NULL;
1325 WCHAR drive[10];
1326 WCHAR dir[MAX_PATH];
1327 WCHAR fname[MAX_PATH];
1328 WCHAR ext[MAX_PATH];
Jason Edmeades32fc3662007-04-11 22:25:51 +01001329 DWORD attribs;
Dave Pickles74f440e1999-06-06 15:24:04 +00001330
Jason Edmeades32fc3662007-04-11 22:25:51 +01001331 errorlevel = 0;
1332
1333 /* Must be at least two args */
Markus Amsler765ff5d2006-10-31 21:11:28 +01001334 if (param1[0] == 0x00 || param2[0] == 0x00) {
Jason Edmeades5cc492c2007-06-03 22:07:39 +01001335 WCMD_output (WCMD_LoadMessage(WCMD_NOARG));
Jason Edmeades32fc3662007-04-11 22:25:51 +01001336 errorlevel = 1;
Markus Amsler765ff5d2006-10-31 21:11:28 +01001337 return;
1338 }
Jason Edmeades32fc3662007-04-11 22:25:51 +01001339
1340 /* Destination cannot contain a drive letter or directory separator */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001341 if ((strchrW(param1,':') != NULL) || (strchrW(param1,'\\') != NULL)) {
Jason Edmeades32fc3662007-04-11 22:25:51 +01001342 SetLastError(ERROR_INVALID_PARAMETER);
1343 WCMD_print_error();
1344 errorlevel = 1;
1345 return;
Dave Pickles74f440e1999-06-06 15:24:04 +00001346 }
Jason Edmeades32fc3662007-04-11 22:25:51 +01001347
1348 /* Convert partial path to full path */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001349 GetFullPathName (param1, sizeof(input)/sizeof(WCHAR), input, NULL);
1350 WINE_TRACE("Rename from '%s'('%s') to '%s'\n", wine_dbgstr_w(input),
1351 wine_dbgstr_w(param1), wine_dbgstr_w(param2));
1352 dotDst = strchrW(param2, '.');
Jason Edmeades32fc3662007-04-11 22:25:51 +01001353
1354 /* Split into components */
1355 WCMD_splitpath(input, drive, dir, fname, ext);
1356
1357 hff = FindFirstFile (input, &fd);
1358 while (hff != INVALID_HANDLE_VALUE) {
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001359 WCHAR dest[MAX_PATH];
1360 WCHAR src[MAX_PATH];
1361 WCHAR *dotSrc = NULL;
Jason Edmeades32fc3662007-04-11 22:25:51 +01001362 int dirLen;
1363
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001364 WINE_TRACE("Processing file '%s'\n", wine_dbgstr_w(fd.cFileName));
Jason Edmeades32fc3662007-04-11 22:25:51 +01001365
1366 /* FIXME: If dest name or extension is *, replace with filename/ext
1367 part otherwise use supplied name. This supports:
1368 ren *.fred *.jim
1369 ren jim.* fred.* etc
1370 However, windows has a more complex algorithum supporting eg
1371 ?'s and *'s mid name */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001372 dotSrc = strchrW(fd.cFileName, '.');
Jason Edmeades32fc3662007-04-11 22:25:51 +01001373
1374 /* Build src & dest name */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001375 strcpyW(src, drive);
1376 strcatW(src, dir);
1377 strcpyW(dest, src);
1378 dirLen = strlenW(src);
1379 strcatW(src, fd.cFileName);
Jason Edmeades32fc3662007-04-11 22:25:51 +01001380
1381 /* Build name */
1382 if (param2[0] == '*') {
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001383 strcatW(dest, fd.cFileName);
Jason Edmeades32fc3662007-04-11 22:25:51 +01001384 if (dotSrc) dest[dirLen + (dotSrc - fd.cFileName)] = 0x00;
1385 } else {
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001386 strcatW(dest, param2);
Jason Edmeades32fc3662007-04-11 22:25:51 +01001387 if (dotDst) dest[dirLen + (dotDst - param2)] = 0x00;
1388 }
1389
1390 /* Build Extension */
1391 if (dotDst && (*(dotDst+1)=='*')) {
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001392 if (dotSrc) strcatW(dest, dotSrc);
Jason Edmeades32fc3662007-04-11 22:25:51 +01001393 } else if (dotDst) {
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001394 if (dotDst) strcatW(dest, dotDst);
Jason Edmeades32fc3662007-04-11 22:25:51 +01001395 }
1396
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001397 WINE_TRACE("Source '%s'\n", wine_dbgstr_w(src));
1398 WINE_TRACE("Dest '%s'\n", wine_dbgstr_w(dest));
Jason Edmeades32fc3662007-04-11 22:25:51 +01001399
1400 /* Check if file is read only, otherwise move it */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001401 attribs = GetFileAttributes(src);
Jason Edmeades32fc3662007-04-11 22:25:51 +01001402 if ((attribs != INVALID_FILE_ATTRIBUTES) &&
1403 (attribs & FILE_ATTRIBUTE_READONLY)) {
1404 SetLastError(ERROR_ACCESS_DENIED);
1405 status = 0;
1406 } else {
1407 status = MoveFile (src, dest);
1408 }
1409
1410 if (!status) {
1411 WCMD_print_error ();
1412 errorlevel = 1;
1413 }
1414
1415 /* Step on to next match */
1416 if (FindNextFile(hff, &fd) == 0) {
1417 FindClose(hff);
1418 hff = INVALID_HANDLE_VALUE;
1419 break;
1420 }
1421 }
Dave Pickles74f440e1999-06-06 15:24:04 +00001422}
1423
1424/*****************************************************************************
Mike McCormack9643b792004-03-22 22:56:58 +00001425 * WCMD_dupenv
1426 *
1427 * Make a copy of the environment.
1428 */
Eric Pouech5c2a8912004-11-29 18:00:10 +00001429static WCHAR *WCMD_dupenv( const WCHAR *env )
Mike McCormack9643b792004-03-22 22:56:58 +00001430{
1431 WCHAR *env_copy;
1432 int len;
1433
1434 if( !env )
1435 return NULL;
1436
1437 len = 0;
1438 while ( env[len] )
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001439 len += (strlenW(&env[len]) + 1);
Mike McCormack9643b792004-03-22 22:56:58 +00001440
1441 env_copy = LocalAlloc (LMEM_FIXED, (len+1) * sizeof (WCHAR) );
1442 if (!env_copy)
1443 {
Jason Edmeades5cc492c2007-06-03 22:07:39 +01001444 WINE_ERR("out of memory\n");
Mike McCormack9643b792004-03-22 22:56:58 +00001445 return env_copy;
1446 }
1447 memcpy (env_copy, env, len*sizeof (WCHAR));
1448 env_copy[len] = 0;
1449
1450 return env_copy;
1451}
1452
1453/*****************************************************************************
1454 * WCMD_setlocal
1455 *
1456 * setlocal pushes the environment onto a stack
1457 * Save the environment as unicode so we don't screw anything up.
1458 */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001459void WCMD_setlocal (const WCHAR *s) {
Mike McCormack9643b792004-03-22 22:56:58 +00001460 WCHAR *env;
1461 struct env_stack *env_copy;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001462 WCHAR cwd[MAX_PATH];
Mike McCormack9643b792004-03-22 22:56:58 +00001463
1464 /* DISABLEEXTENSIONS ignored */
1465
1466 env_copy = LocalAlloc (LMEM_FIXED, sizeof (struct env_stack));
1467 if( !env_copy )
1468 {
Jason Edmeades5cc492c2007-06-03 22:07:39 +01001469 WINE_ERR ("out of memory\n");
Mike McCormack9643b792004-03-22 22:56:58 +00001470 return;
1471 }
1472
1473 env = GetEnvironmentStringsW ();
1474
1475 env_copy->strings = WCMD_dupenv (env);
1476 if (env_copy->strings)
1477 {
1478 env_copy->next = saved_environment;
1479 saved_environment = env_copy;
Jason Edmeades86140e32007-03-08 00:48:49 +00001480
1481 /* Save the current drive letter */
1482 GetCurrentDirectory (MAX_PATH, cwd);
Francois Gouget540d8182007-03-12 10:35:13 +01001483 env_copy->u.cwd = cwd[0];
Mike McCormack9643b792004-03-22 22:56:58 +00001484 }
1485 else
1486 LocalFree (env_copy);
1487
1488 FreeEnvironmentStringsW (env);
Jason Edmeades86140e32007-03-08 00:48:49 +00001489
Mike McCormack9643b792004-03-22 22:56:58 +00001490}
1491
1492/*****************************************************************************
Mike McCormack9643b792004-03-22 22:56:58 +00001493 * WCMD_endlocal
1494 *
1495 * endlocal pops the environment off a stack
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001496 * Note: When searching for '=', search from WCHAR position 1, to handle
Jason Edmeades86140e32007-03-08 00:48:49 +00001497 * special internal environment variables =C:, =D: etc
Mike McCormack9643b792004-03-22 22:56:58 +00001498 */
1499void WCMD_endlocal (void) {
1500 WCHAR *env, *old, *p;
1501 struct env_stack *temp;
1502 int len, n;
1503
1504 if (!saved_environment)
1505 return;
1506
1507 /* pop the old environment from the stack */
1508 temp = saved_environment;
1509 saved_environment = temp->next;
1510
1511 /* delete the current environment, totally */
1512 env = GetEnvironmentStringsW ();
1513 old = WCMD_dupenv (GetEnvironmentStringsW ());
1514 len = 0;
1515 while (old[len]) {
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001516 n = strlenW(&old[len]) + 1;
1517 p = strchrW(&old[len] + 1, '=');
Mike McCormack9643b792004-03-22 22:56:58 +00001518 if (p)
1519 {
1520 *p++ = 0;
1521 SetEnvironmentVariableW (&old[len], NULL);
1522 }
1523 len += n;
1524 }
1525 LocalFree (old);
1526 FreeEnvironmentStringsW (env);
Jason Edmeadesfda72292007-02-27 23:18:57 +00001527
Mike McCormack9643b792004-03-22 22:56:58 +00001528 /* restore old environment */
1529 env = temp->strings;
1530 len = 0;
1531 while (env[len]) {
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001532 n = strlenW(&env[len]) + 1;
1533 p = strchrW(&env[len] + 1, '=');
Mike McCormack9643b792004-03-22 22:56:58 +00001534 if (p)
1535 {
1536 *p++ = 0;
1537 SetEnvironmentVariableW (&env[len], p);
1538 }
1539 len += n;
1540 }
Jason Edmeades86140e32007-03-08 00:48:49 +00001541
1542 /* Restore current drive letter */
Francois Gouget540d8182007-03-12 10:35:13 +01001543 if (IsCharAlpha(temp->u.cwd)) {
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001544 WCHAR envvar[4];
1545 WCHAR cwd[MAX_PATH];
1546 static const WCHAR fmt[] = {'=','%','c',':','\0'};
1547
1548 wsprintf(envvar, fmt, temp->u.cwd);
Jason Edmeades86140e32007-03-08 00:48:49 +00001549 if (GetEnvironmentVariable(envvar, cwd, MAX_PATH)) {
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001550 WINE_TRACE("Resetting cwd to %s\n", wine_dbgstr_w(cwd));
Jason Edmeades86140e32007-03-08 00:48:49 +00001551 SetCurrentDirectory(cwd);
1552 }
1553 }
1554
Mike McCormack9643b792004-03-22 22:56:58 +00001555 LocalFree (env);
1556 LocalFree (temp);
1557}
1558
1559/*****************************************************************************
Dave Pickles74f440e1999-06-06 15:24:04 +00001560 * WCMD_setshow_attrib
1561 *
1562 * Display and optionally sets DOS attributes on a file or directory
1563 *
1564 * FIXME: Wine currently uses the Unix stat() function to get file attributes.
1565 * As a result only the Readonly flag is correctly reported, the Archive bit
1566 * is always set and the rest are not implemented. We do the Right Thing anyway.
1567 *
Dave Pickles036a9f71999-07-10 11:36:45 +00001568 * FIXME: No SET functionality.
1569 *
Dave Pickles74f440e1999-06-06 15:24:04 +00001570 */
1571
Eric Pouechb2f079b2003-02-27 01:41:21 +00001572void WCMD_setshow_attrib (void) {
Dave Pickles74f440e1999-06-06 15:24:04 +00001573
Alexandre Julliardabfe1052007-03-01 12:43:19 +01001574 DWORD count;
1575 HANDLE hff;
1576 WIN32_FIND_DATA fd;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001577 WCHAR flags[9] = {' ',' ',' ',' ',' ',' ',' ',' ','\0'};
Dave Pickles74f440e1999-06-06 15:24:04 +00001578
Dave Pickles036a9f71999-07-10 11:36:45 +00001579 if (param1[0] == '-') {
Jason Edmeades5cc492c2007-06-03 22:07:39 +01001580 WCMD_output (WCMD_LoadMessage(WCMD_NYI));
Dave Pickles036a9f71999-07-10 11:36:45 +00001581 return;
1582 }
1583
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001584 if (strlenW(param1) == 0) {
1585 static const WCHAR slashStarW[] = {'\\','*','\0'};
1586
1587 GetCurrentDirectory (sizeof(param1)/sizeof(WCHAR), param1);
1588 strcatW (param1, slashStarW);
Dave Pickles74f440e1999-06-06 15:24:04 +00001589 }
1590
1591 hff = FindFirstFile (param1, &fd);
1592 if (hff == INVALID_HANDLE_VALUE) {
Jason Edmeades5cc492c2007-06-03 22:07:39 +01001593 WCMD_output (WCMD_LoadMessage(WCMD_FILENOTFOUND), param1);
Dave Pickles74f440e1999-06-06 15:24:04 +00001594 }
1595 else {
1596 do {
1597 if (!(fd.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY)) {
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001598 static const WCHAR fmt[] = {'%','s',' ',' ',' ','%','s','\n','\0'};
Dave Pickles74f440e1999-06-06 15:24:04 +00001599 if (fd.dwFileAttributes & FILE_ATTRIBUTE_HIDDEN) {
1600 flags[0] = 'H';
1601 }
1602 if (fd.dwFileAttributes & FILE_ATTRIBUTE_SYSTEM) {
1603 flags[1] = 'S';
1604 }
1605 if (fd.dwFileAttributes & FILE_ATTRIBUTE_ARCHIVE) {
1606 flags[2] = 'A';
1607 }
1608 if (fd.dwFileAttributes & FILE_ATTRIBUTE_READONLY) {
1609 flags[3] = 'R';
1610 }
1611 if (fd.dwFileAttributes & FILE_ATTRIBUTE_TEMPORARY) {
1612 flags[4] = 'T';
1613 }
1614 if (fd.dwFileAttributes & FILE_ATTRIBUTE_COMPRESSED) {
1615 flags[5] = 'C';
1616 }
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001617 WCMD_output (fmt, flags, fd.cFileName);
Dave Pickles74f440e1999-06-06 15:24:04 +00001618 for (count=0; count < 8; count++) flags[count] = ' ';
1619 }
1620 } while (FindNextFile(hff, &fd) != 0);
1621 }
1622 FindClose (hff);
1623}
1624
1625/*****************************************************************************
1626 * WCMD_setshow_default
1627 *
1628 * Set/Show the current default directory
1629 */
1630
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001631void WCMD_setshow_default (WCHAR *command) {
Dave Pickles74f440e1999-06-06 15:24:04 +00001632
Alexandre Julliardabfe1052007-03-01 12:43:19 +01001633 BOOL status;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001634 WCHAR string[1024];
1635 WCHAR cwd[1024];
1636 WCHAR *pos;
Jason Edmeades5c8c6942007-03-08 00:43:09 +00001637 WIN32_FIND_DATA fd;
1638 HANDLE hff;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001639 static const WCHAR parmD[] = {'/','D','\0'};
Dave Pickles74f440e1999-06-06 15:24:04 +00001640
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001641 WINE_TRACE("Request change to directory '%s'\n", wine_dbgstr_w(command));
Jason Edmeadesffbaede2007-03-08 00:50:14 +00001642
1643 /* Skip /D and trailing whitespace if on the front of the command line */
1644 if (CompareString (LOCALE_USER_DEFAULT,
1645 NORM_IGNORECASE | SORT_STRINGSORT,
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001646 command, 2, parmD, -1) == 2) {
Jason Edmeadesffbaede2007-03-08 00:50:14 +00001647 command += 2;
1648 while (*command && *command==' ') command++;
1649 }
1650
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001651 GetCurrentDirectory (sizeof(cwd)/sizeof(WCHAR), cwd);
1652 if (strlenW(command) == 0) {
1653 strcatW (cwd, newline);
Jason Edmeadesffbaede2007-03-08 00:50:14 +00001654 WCMD_output (cwd);
Dave Pickles74f440e1999-06-06 15:24:04 +00001655 }
1656 else {
Jason Edmeadeseb6d0842007-03-08 00:42:33 +00001657 /* Remove any double quotes, which may be in the
1658 middle, eg. cd "C:\Program Files"\Microsoft is ok */
1659 pos = string;
1660 while (*command) {
1661 if (*command != '"') *pos++ = *command;
1662 command++;
1663 }
1664 *pos = 0x00;
1665
Jason Edmeades5c8c6942007-03-08 00:43:09 +00001666 /* Search for approprate directory */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001667 WINE_TRACE("Looking for directory '%s'\n", wine_dbgstr_w(string));
Jason Edmeades5c8c6942007-03-08 00:43:09 +00001668 hff = FindFirstFile (string, &fd);
1669 while (hff != INVALID_HANDLE_VALUE) {
1670 if (fd.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) {
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001671 WCHAR fpath[MAX_PATH];
1672 WCHAR drive[10];
1673 WCHAR dir[MAX_PATH];
1674 WCHAR fname[MAX_PATH];
1675 WCHAR ext[MAX_PATH];
1676 static const WCHAR fmt[] = {'%','s','%','s','%','s','\0'};
Jason Edmeades5c8c6942007-03-08 00:43:09 +00001677
1678 /* Convert path into actual directory spec */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001679 GetFullPathName (string, sizeof(fpath)/sizeof(WCHAR), fpath, NULL);
Jason Edmeades5c8c6942007-03-08 00:43:09 +00001680 WCMD_splitpath(fpath, drive, dir, fname, ext);
1681
1682 /* Rebuild path */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001683 wsprintf(string, fmt, drive, dir, fd.cFileName);
Jason Edmeades5c8c6942007-03-08 00:43:09 +00001684
1685 FindClose(hff);
1686 hff = INVALID_HANDLE_VALUE;
1687 break;
1688 }
1689
1690 /* Step on to next match */
1691 if (FindNextFile(hff, &fd) == 0) {
1692 FindClose(hff);
1693 hff = INVALID_HANDLE_VALUE;
1694 break;
1695 }
1696 }
1697
Jason Edmeadeseb6d0842007-03-08 00:42:33 +00001698 /* Change to that directory */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001699 WINE_TRACE("Really changing to directory '%s'\n", wine_dbgstr_w(string));
Jason Edmeades2b03d7d2007-03-08 00:47:45 +00001700
Jason Edmeadeseb6d0842007-03-08 00:42:33 +00001701 status = SetCurrentDirectory (string);
Dave Pickles74f440e1999-06-06 15:24:04 +00001702 if (!status) {
Jason Edmeades121a8302007-03-08 00:43:32 +00001703 errorlevel = 1;
Dave Pickles74f440e1999-06-06 15:24:04 +00001704 WCMD_print_error ();
1705 return;
Jason Edmeadesffbaede2007-03-08 00:50:14 +00001706 } else {
1707
1708 /* Restore old directory if drive letter would change, and
1709 CD x:\directory /D (or pushd c:\directory) not supplied */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001710 if ((strstrW(quals, parmD) == NULL) &&
Jason Edmeadesffbaede2007-03-08 00:50:14 +00001711 (param1[1] == ':') && (toupper(param1[0]) != toupper(cwd[0]))) {
1712 SetCurrentDirectory(cwd);
1713 }
Dave Pickles74f440e1999-06-06 15:24:04 +00001714 }
Jason Edmeades2b03d7d2007-03-08 00:47:45 +00001715
Jason Edmeadesffbaede2007-03-08 00:50:14 +00001716 /* Set special =C: type environment variable, for drive letter of
1717 change of directory, even if path was restored due to missing
1718 /D (allows changing drive letter when not resident on that
1719 drive */
Jason Edmeades2b03d7d2007-03-08 00:47:45 +00001720 if ((string[1] == ':') && IsCharAlpha (string[0])) {
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001721 WCHAR env[4];
1722 strcpyW(env, equalW);
1723 memcpy(env+1, string, 2 * sizeof(WCHAR));
Jason Edmeades2b03d7d2007-03-08 00:47:45 +00001724 env[3] = 0x00;
Alexandre Julliardabc5fef2007-06-08 12:54:21 +02001725 WINE_TRACE("Setting '%s' to '%s'\n", wine_dbgstr_w(env), wine_dbgstr_w(string));
Jason Edmeades2b03d7d2007-03-08 00:47:45 +00001726 SetEnvironmentVariable(env, string);
1727 }
1728
Dave Pickles74f440e1999-06-06 15:24:04 +00001729 }
1730 return;
1731}
1732
1733/****************************************************************************
1734 * WCMD_setshow_date
1735 *
1736 * Set/Show the system date
1737 * FIXME: Can't change date yet
1738 */
1739
Eric Pouechb2f079b2003-02-27 01:41:21 +00001740void WCMD_setshow_date (void) {
Dave Pickles74f440e1999-06-06 15:24:04 +00001741
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001742 WCHAR curdate[64], buffer[64];
Alexandre Julliardabfe1052007-03-01 12:43:19 +01001743 DWORD count;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001744 static const WCHAR parmT[] = {'/','T','\0'};
Dave Pickles74f440e1999-06-06 15:24:04 +00001745
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001746 if (strlenW(param1) == 0) {
Dave Pickles74f440e1999-06-06 15:24:04 +00001747 if (GetDateFormat (LOCALE_USER_DEFAULT, 0, NULL, NULL,
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001748 curdate, sizeof(curdate)/sizeof(WCHAR))) {
Jason Edmeades5cc492c2007-06-03 22:07:39 +01001749 WCMD_output (WCMD_LoadMessage(WCMD_CURRENTDATE), curdate);
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001750 if (strstrW (quals, parmT) == NULL) {
Jason Edmeades5cc492c2007-06-03 22:07:39 +01001751 WCMD_output (WCMD_LoadMessage(WCMD_NEWDATE));
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001752 WCMD_ReadFile (GetStdHandle(STD_INPUT_HANDLE),
1753 buffer, sizeof(buffer)/sizeof(WCHAR), &count, NULL);
Jason Edmeades988ef412007-03-08 00:45:05 +00001754 if (count > 2) {
Jason Edmeades5cc492c2007-06-03 22:07:39 +01001755 WCMD_output (WCMD_LoadMessage(WCMD_NYI));
Jason Edmeades988ef412007-03-08 00:45:05 +00001756 }
Dave Pickles74f440e1999-06-06 15:24:04 +00001757 }
1758 }
1759 else WCMD_print_error ();
1760 }
1761 else {
Jason Edmeades5cc492c2007-06-03 22:07:39 +01001762 WCMD_output (WCMD_LoadMessage(WCMD_NYI));
Dave Pickles74f440e1999-06-06 15:24:04 +00001763 }
1764}
1765
1766/****************************************************************************
Mike McCormackbbe0e2c2003-12-30 19:17:31 +00001767 * WCMD_compare
1768 */
Eric Pouech5c2a8912004-11-29 18:00:10 +00001769static int WCMD_compare( const void *a, const void *b )
Mike McCormackbbe0e2c2003-12-30 19:17:31 +00001770{
1771 int r;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001772 const WCHAR * const *str_a = a, * const *str_b = b;
Mike McCormackbbe0e2c2003-12-30 19:17:31 +00001773 r = CompareString( LOCALE_USER_DEFAULT, NORM_IGNORECASE | SORT_STRINGSORT,
1774 *str_a, -1, *str_b, -1 );
1775 if( r == CSTR_LESS_THAN ) return -1;
1776 if( r == CSTR_GREATER_THAN ) return 1;
1777 return 0;
1778}
1779
1780/****************************************************************************
1781 * WCMD_setshow_sortenv
1782 *
1783 * sort variables into order for display
Jason Edmeades1c632cd2007-02-26 23:06:11 +00001784 * Optionally only display those who start with a stub
1785 * returns the count displayed
Mike McCormackbbe0e2c2003-12-30 19:17:31 +00001786 */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001787static int WCMD_setshow_sortenv(const WCHAR *s, const WCHAR *stub)
Mike McCormackbbe0e2c2003-12-30 19:17:31 +00001788{
Jason Edmeades1c632cd2007-02-26 23:06:11 +00001789 UINT count=0, len=0, i, displayedcount=0, stublen=0;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001790 const WCHAR **str;
Mike McCormackbbe0e2c2003-12-30 19:17:31 +00001791
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001792 if (stub) stublen = strlenW(stub);
Jason Edmeades1c632cd2007-02-26 23:06:11 +00001793
Mike McCormackbbe0e2c2003-12-30 19:17:31 +00001794 /* count the number of strings, and the total length */
1795 while ( s[len] ) {
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001796 len += (strlenW(&s[len]) + 1);
Mike McCormackbbe0e2c2003-12-30 19:17:31 +00001797 count++;
1798 }
1799
1800 /* add the strings to an array */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001801 str = LocalAlloc (LMEM_FIXED | LMEM_ZEROINIT, count * sizeof (WCHAR*) );
Mike McCormackbbe0e2c2003-12-30 19:17:31 +00001802 if( !str )
Jason Edmeades1c632cd2007-02-26 23:06:11 +00001803 return 0;
Mike McCormackbbe0e2c2003-12-30 19:17:31 +00001804 str[0] = s;
1805 for( i=1; i<count; i++ )
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001806 str[i] = str[i-1] + strlenW(str[i-1]) + 1;
Mike McCormackbbe0e2c2003-12-30 19:17:31 +00001807
1808 /* sort the array */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001809 qsort( str, count, sizeof (WCHAR*), WCMD_compare );
Mike McCormackbbe0e2c2003-12-30 19:17:31 +00001810
1811 /* print it */
Rein Klazesa18ea3d2005-12-01 15:58:16 +01001812 for( i=0; i<count; i++ ) {
Jason Edmeades1c632cd2007-02-26 23:06:11 +00001813 if (!stub || CompareString (LOCALE_USER_DEFAULT,
1814 NORM_IGNORECASE | SORT_STRINGSORT,
1815 str[i], stublen, stub, -1) == 2) {
Jason Edmeadesd1317f52007-03-08 00:48:17 +00001816 /* Don't display special internal variables */
1817 if (str[i][0] != '=') {
1818 WCMD_output_asis(str[i]);
Jason Edmeades5cc492c2007-06-03 22:07:39 +01001819 WCMD_output_asis(newline);
Jason Edmeadesd1317f52007-03-08 00:48:17 +00001820 displayedcount++;
1821 }
Jason Edmeades1c632cd2007-02-26 23:06:11 +00001822 }
Rein Klazesa18ea3d2005-12-01 15:58:16 +01001823 }
Mike McCormackbbe0e2c2003-12-30 19:17:31 +00001824
1825 LocalFree( str );
Jason Edmeades1c632cd2007-02-26 23:06:11 +00001826 return displayedcount;
Mike McCormackbbe0e2c2003-12-30 19:17:31 +00001827}
1828
1829/****************************************************************************
Dave Pickles74f440e1999-06-06 15:24:04 +00001830 * WCMD_setshow_env
1831 *
1832 * Set/Show the environment variables
Dave Pickles74f440e1999-06-06 15:24:04 +00001833 */
1834
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001835void WCMD_setshow_env (WCHAR *s) {
Dave Pickles74f440e1999-06-06 15:24:04 +00001836
Alexandre Julliardabfe1052007-03-01 12:43:19 +01001837 LPVOID env;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001838 WCHAR *p;
Alexandre Julliardabfe1052007-03-01 12:43:19 +01001839 int status;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001840 static const WCHAR parmP[] = {'/','P','\0'};
Dave Pickles74f440e1999-06-06 15:24:04 +00001841
Jason Edmeades87f02932007-03-13 00:11:31 +00001842 errorlevel = 0;
Jason Edmeades4b1ef912007-03-13 00:11:04 +00001843 if (param1[0] == 0x00 && quals[0] == 0x00) {
Dave Pickles74f440e1999-06-06 15:24:04 +00001844 env = GetEnvironmentStrings ();
Jason Edmeades1c632cd2007-02-26 23:06:11 +00001845 WCMD_setshow_sortenv( env, NULL );
Jason Edmeades4b1ef912007-03-13 00:11:04 +00001846 return;
Dave Pickles74f440e1999-06-06 15:24:04 +00001847 }
Jason Edmeades4b1ef912007-03-13 00:11:04 +00001848
1849 /* See if /P supplied, and if so echo the prompt, and read in a reply */
1850 if (CompareString (LOCALE_USER_DEFAULT,
1851 NORM_IGNORECASE | SORT_STRINGSORT,
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001852 s, 2, parmP, -1) == 2) {
1853 WCHAR string[MAXSTRING];
Jason Edmeades4b1ef912007-03-13 00:11:04 +00001854 DWORD count;
1855
1856 s += 2;
1857 while (*s && *s==' ') s++;
1858
1859 /* If no parameter, or no '=' sign, return an error */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001860 if (!(*s) || ((p = strchrW (s, '=')) == NULL )) {
Jason Edmeades5cc492c2007-06-03 22:07:39 +01001861 WCMD_output (WCMD_LoadMessage(WCMD_NOARG));
Jason Edmeades4b1ef912007-03-13 00:11:04 +00001862 return;
1863 }
1864
1865 /* Output the prompt */
1866 *p++ = '\0';
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001867 if (strlenW(p) != 0) WCMD_output(p);
Jason Edmeades4b1ef912007-03-13 00:11:04 +00001868
1869 /* Read the reply */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001870 WCMD_ReadFile (GetStdHandle(STD_INPUT_HANDLE), string,
1871 sizeof(string)/sizeof(WCHAR), &count, NULL);
Jason Edmeades4b1ef912007-03-13 00:11:04 +00001872 if (count > 1) {
1873 string[count-1] = '\0'; /* ReadFile output is not null-terminated! */
1874 if (string[count-2] == '\r') string[count-2] = '\0'; /* Under Windoze we get CRLF! */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001875 WINE_TRACE("set /p: Setting var '%s' to '%s'\n", wine_dbgstr_w(s),
1876 wine_dbgstr_w(string));
Jason Edmeades4b1ef912007-03-13 00:11:04 +00001877 status = SetEnvironmentVariable (s, string);
1878 }
1879
1880 } else {
Jason Edmeades87f02932007-03-13 00:11:31 +00001881 DWORD gle;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001882 p = strchrW (s, '=');
Dave Pickles74f440e1999-06-06 15:24:04 +00001883 if (p == NULL) {
Jason Edmeades1c632cd2007-02-26 23:06:11 +00001884 env = GetEnvironmentStrings ();
1885 if (WCMD_setshow_sortenv( env, s ) == 0) {
Jason Edmeades5cc492c2007-06-03 22:07:39 +01001886 WCMD_output (WCMD_LoadMessage(WCMD_MISSINGENV), s);
Jason Edmeades87f02932007-03-13 00:11:31 +00001887 errorlevel = 1;
Vincent Béron9a624912002-05-31 23:06:46 +00001888 }
Dave Pickles74f440e1999-06-06 15:24:04 +00001889 return;
1890 }
1891 *p++ = '\0';
Jason Edmeadesa5910f42000-08-01 02:14:33 +00001892
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001893 if (strlenW(p) == 0) p = NULL;
Vincent Béron9a624912002-05-31 23:06:46 +00001894 status = SetEnvironmentVariable (s, p);
Jason Edmeades87f02932007-03-13 00:11:31 +00001895 gle = GetLastError();
1896 if ((!status) & (gle == ERROR_ENVVAR_NOT_FOUND)) {
1897 errorlevel = 1;
1898 } else if ((!status)) WCMD_print_error();
Dave Pickles74f440e1999-06-06 15:24:04 +00001899 }
Dave Pickles74f440e1999-06-06 15:24:04 +00001900}
1901
1902/****************************************************************************
1903 * WCMD_setshow_path
1904 *
1905 * Set/Show the path environment variable
1906 */
1907
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001908void WCMD_setshow_path (WCHAR *command) {
Dave Pickles74f440e1999-06-06 15:24:04 +00001909
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001910 WCHAR string[1024];
Alexandre Julliardabfe1052007-03-01 12:43:19 +01001911 DWORD status;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001912 static const WCHAR pathW[] = {'P','A','T','H','\0'};
1913 static const WCHAR pathEqW[] = {'P','A','T','H','=','\0'};
Dave Pickles74f440e1999-06-06 15:24:04 +00001914
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001915 if (strlenW(param1) == 0) {
1916 status = GetEnvironmentVariable (pathW, string, sizeof(string)/sizeof(WCHAR));
Dave Pickles74f440e1999-06-06 15:24:04 +00001917 if (status != 0) {
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001918 WCMD_output_asis ( pathEqW);
Rein Klazes0bf64a42005-12-02 11:25:51 +01001919 WCMD_output_asis ( string);
Jason Edmeades5cc492c2007-06-03 22:07:39 +01001920 WCMD_output_asis ( newline);
Dave Pickles74f440e1999-06-06 15:24:04 +00001921 }
1922 else {
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001923 WCMD_output (WCMD_LoadMessage(WCMD_NOPATH));
Dave Pickles74f440e1999-06-06 15:24:04 +00001924 }
1925 }
1926 else {
Jason Edmeades73587982007-02-20 00:37:47 +00001927 if (*command == '=') command++; /* Skip leading '=' */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001928 status = SetEnvironmentVariable (pathW, command);
Dave Pickles74f440e1999-06-06 15:24:04 +00001929 if (!status) WCMD_print_error();
1930 }
1931}
1932
1933/****************************************************************************
1934 * WCMD_setshow_prompt
1935 *
1936 * Set or show the command prompt.
1937 */
1938
Eric Pouechb2f079b2003-02-27 01:41:21 +00001939void WCMD_setshow_prompt (void) {
Dave Pickles74f440e1999-06-06 15:24:04 +00001940
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001941 WCHAR *s;
1942 static const WCHAR promptW[] = {'P','R','O','M','P','T','\0'};
Dave Pickles74f440e1999-06-06 15:24:04 +00001943
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001944 if (strlenW(param1) == 0) {
1945 SetEnvironmentVariable (promptW, NULL);
Dave Pickles74f440e1999-06-06 15:24:04 +00001946 }
1947 else {
1948 s = param1;
1949 while ((*s == '=') || (*s == ' ')) s++;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001950 if (strlenW(s) == 0) {
1951 SetEnvironmentVariable (promptW, NULL);
Dave Pickles74f440e1999-06-06 15:24:04 +00001952 }
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001953 else SetEnvironmentVariable (promptW, s);
Dave Pickles74f440e1999-06-06 15:24:04 +00001954 }
1955}
1956
1957/****************************************************************************
1958 * WCMD_setshow_time
1959 *
1960 * Set/Show the system time
1961 * FIXME: Can't change time yet
1962 */
1963
Eric Pouechb2f079b2003-02-27 01:41:21 +00001964void WCMD_setshow_time (void) {
Dave Pickles74f440e1999-06-06 15:24:04 +00001965
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001966 WCHAR curtime[64], buffer[64];
Alexandre Julliardabfe1052007-03-01 12:43:19 +01001967 DWORD count;
1968 SYSTEMTIME st;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001969 static const WCHAR parmT[] = {'/','T','\0'};
Dave Pickles74f440e1999-06-06 15:24:04 +00001970
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001971 if (strlenW(param1) == 0) {
Dave Pickles036a9f71999-07-10 11:36:45 +00001972 GetLocalTime(&st);
1973 if (GetTimeFormat (LOCALE_USER_DEFAULT, 0, &st, NULL,
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001974 curtime, sizeof(curtime)/sizeof(WCHAR))) {
Jason Edmeades5cc492c2007-06-03 22:07:39 +01001975 WCMD_output (WCMD_LoadMessage(WCMD_CURRENTDATE), curtime);
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001976 if (strstrW (quals, parmT) == NULL) {
Jason Edmeades5cc492c2007-06-03 22:07:39 +01001977 WCMD_output (WCMD_LoadMessage(WCMD_NEWTIME));
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001978 WCMD_ReadFile (GetStdHandle(STD_INPUT_HANDLE), buffer,
1979 sizeof(buffer)/sizeof(WCHAR), &count, NULL);
Jason Edmeadesb6ed73b2007-03-08 00:45:27 +00001980 if (count > 2) {
Jason Edmeades5cc492c2007-06-03 22:07:39 +01001981 WCMD_output (WCMD_LoadMessage(WCMD_NYI));
Jason Edmeadesb6ed73b2007-03-08 00:45:27 +00001982 }
Dave Pickles74f440e1999-06-06 15:24:04 +00001983 }
1984 }
1985 else WCMD_print_error ();
1986 }
1987 else {
Jason Edmeades5cc492c2007-06-03 22:07:39 +01001988 WCMD_output (WCMD_LoadMessage(WCMD_NYI));
Dave Pickles74f440e1999-06-06 15:24:04 +00001989 }
1990}
1991
1992/****************************************************************************
1993 * WCMD_shift
1994 *
1995 * Shift batch parameters.
Jason Edmeades612dc9d2007-03-08 00:50:37 +00001996 * Optional /n says where to start shifting (n=0-8)
Dave Pickles74f440e1999-06-06 15:24:04 +00001997 */
1998
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01001999void WCMD_shift (WCHAR *command) {
Jason Edmeades612dc9d2007-03-08 00:50:37 +00002000 int start;
Dave Pickles74f440e1999-06-06 15:24:04 +00002001
Jason Edmeades612dc9d2007-03-08 00:50:37 +00002002 if (context != NULL) {
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002003 WCHAR *pos = strchrW(command, '/');
Jason Edmeades612dc9d2007-03-08 00:50:37 +00002004 int i;
2005
2006 if (pos == NULL) {
2007 start = 0;
2008 } else if (*(pos+1)>='0' && *(pos+1)<='8') {
2009 start = (*(pos+1) - '0');
2010 } else {
2011 SetLastError(ERROR_INVALID_PARAMETER);
2012 WCMD_print_error();
2013 return;
2014 }
2015
2016 WINE_TRACE("Shifting variables, starting at %d\n", start);
2017 for (i=start;i<=8;i++) {
2018 context -> shift_count[i] = context -> shift_count[i+1] + 1;
2019 }
2020 context -> shift_count[9] = context -> shift_count[9] + 1;
2021 }
Dave Pickles74f440e1999-06-06 15:24:04 +00002022
2023}
2024
2025/****************************************************************************
Jason Edmeadesbcc62562002-05-04 18:29:31 +00002026 * WCMD_title
2027 *
2028 * Set the console title
2029 */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002030void WCMD_title (WCHAR *command) {
Jason Edmeadesbcc62562002-05-04 18:29:31 +00002031 SetConsoleTitle(command);
2032}
2033
2034/****************************************************************************
Dave Pickles74f440e1999-06-06 15:24:04 +00002035 * WCMD_type
2036 *
2037 * Copy a file to standard output.
2038 */
2039
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002040void WCMD_type (WCHAR *command) {
Dave Pickles74f440e1999-06-06 15:24:04 +00002041
Jason Edmeades4ef2f8b2007-03-13 00:09:33 +00002042 int argno = 0;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002043 WCHAR *argN = command;
Jason Edmeades4ef2f8b2007-03-13 00:09:33 +00002044 BOOL writeHeaders = FALSE;
Dave Pickles74f440e1999-06-06 15:24:04 +00002045
Markus Amsler765ff5d2006-10-31 21:11:28 +01002046 if (param1[0] == 0x00) {
Jason Edmeades5cc492c2007-06-03 22:07:39 +01002047 WCMD_output (WCMD_LoadMessage(WCMD_NOARG));
Markus Amsler765ff5d2006-10-31 21:11:28 +01002048 return;
2049 }
Jason Edmeades4ef2f8b2007-03-13 00:09:33 +00002050
2051 if (param2[0] != 0x00) writeHeaders = TRUE;
2052
2053 /* Loop through all args */
2054 errorlevel = 0;
2055 while (argN) {
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002056 WCHAR *thisArg = WCMD_parameter (command, argno++, &argN);
Jason Edmeades4ef2f8b2007-03-13 00:09:33 +00002057
2058 HANDLE h;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002059 WCHAR buffer[512];
Jason Edmeades4ef2f8b2007-03-13 00:09:33 +00002060 DWORD count;
2061
2062 if (!argN) break;
2063
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002064 WINE_TRACE("type: Processing arg '%s'\n", wine_dbgstr_w(thisArg));
Jason Edmeades4ef2f8b2007-03-13 00:09:33 +00002065 h = CreateFile (thisArg, GENERIC_READ, FILE_SHARE_READ, NULL, OPEN_EXISTING,
2066 FILE_ATTRIBUTE_NORMAL, NULL);
2067 if (h == INVALID_HANDLE_VALUE) {
2068 WCMD_print_error ();
Jason Edmeades5cc492c2007-06-03 22:07:39 +01002069 WCMD_output (WCMD_LoadMessage(WCMD_READFAIL), thisArg);
Jason Edmeades4ef2f8b2007-03-13 00:09:33 +00002070 errorlevel = 1;
2071 } else {
2072 if (writeHeaders) {
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002073 static const WCHAR fmt[] = {'\n','%','s','\n','\n','\0'};
2074 WCMD_output(fmt, thisArg);
Jason Edmeades4ef2f8b2007-03-13 00:09:33 +00002075 }
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002076 while (WCMD_ReadFile (h, buffer, sizeof(buffer)/sizeof(WCHAR), &count, NULL)) {
Jason Edmeades4ef2f8b2007-03-13 00:09:33 +00002077 if (count == 0) break; /* ReadFile reports success on EOF! */
2078 buffer[count] = 0;
2079 WCMD_output_asis (buffer);
2080 }
2081 CloseHandle (h);
2082 }
Dave Pickles74f440e1999-06-06 15:24:04 +00002083 }
Dave Pickles74f440e1999-06-06 15:24:04 +00002084}
2085
2086/****************************************************************************
Jason Edmeadesce875222007-04-13 21:34:05 +01002087 * WCMD_more
2088 *
2089 * Output either a file or stdin to screen in pages
2090 */
2091
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002092void WCMD_more (WCHAR *command) {
Jason Edmeadesce875222007-04-13 21:34:05 +01002093
2094 int argno = 0;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002095 WCHAR *argN = command;
Jason Edmeadesce875222007-04-13 21:34:05 +01002096 BOOL useinput = FALSE;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002097 WCHAR moreStr[100];
2098 WCHAR moreStrPage[100];
2099 WCHAR buffer[512];
Jason Edmeadesce875222007-04-13 21:34:05 +01002100 DWORD count;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002101 static const WCHAR moreStart[] = {'-','-',' ','\0'};
2102 static const WCHAR moreFmt[] = {'%','s',' ','-','-','\n','\0'};
2103 static const WCHAR moreFmt2[] = {'%','s',' ','(','%','2','.','2','d','%','%',
2104 ')',' ','-','-','\n','\0'};
2105 static const WCHAR conInW[] = {'C','O','N','I','N','$','\0'};
Jason Edmeadesce875222007-04-13 21:34:05 +01002106
2107 /* Prefix the NLS more with '-- ', then load the text */
2108 errorlevel = 0;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002109 strcpyW(moreStr, moreStart);
2110 LoadString (hinst, WCMD_MORESTR, &moreStr[3],
2111 (sizeof(moreStr)/sizeof(WCHAR))-3);
Jason Edmeadesce875222007-04-13 21:34:05 +01002112
2113 if (param1[0] == 0x00) {
2114
2115 /* Wine implements pipes via temporary files, and hence stdin is
2116 effectively reading from the file. This means the prompts for
2117 more are satistied by the next line from the input (file). To
2118 avoid this, ensure stdin is to the console */
2119 HANDLE hstdin = GetStdHandle(STD_INPUT_HANDLE);
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002120 HANDLE hConIn = CreateFile(conInW, GENERIC_READ | GENERIC_WRITE,
Jason Edmeadesce875222007-04-13 21:34:05 +01002121 FILE_SHARE_READ, NULL, OPEN_EXISTING,
2122 FILE_ATTRIBUTE_NORMAL, 0);
2123 SetStdHandle(STD_INPUT_HANDLE, hConIn);
2124
2125 /* Warning: No easy way of ending the stream (ctrl+z on windows) so
2126 once you get in this bit unless due to a pipe, its going to end badly... */
2127 useinput = TRUE;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002128 wsprintf(moreStrPage, moreFmt, moreStr);
Jason Edmeadesce875222007-04-13 21:34:05 +01002129
2130 WCMD_enter_paged_mode(moreStrPage);
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002131 while (WCMD_ReadFile (hstdin, buffer, (sizeof(buffer)/sizeof(WCHAR))-1, &count, NULL)) {
Jason Edmeadesce875222007-04-13 21:34:05 +01002132 if (count == 0) break; /* ReadFile reports success on EOF! */
2133 buffer[count] = 0;
2134 WCMD_output_asis (buffer);
2135 }
2136 WCMD_leave_paged_mode();
2137
2138 /* Restore stdin to what it was */
2139 SetStdHandle(STD_INPUT_HANDLE, hstdin);
2140 CloseHandle(hConIn);
2141
2142 return;
2143 } else {
2144 BOOL needsPause = FALSE;
2145
2146 /* Loop through all args */
2147 WCMD_enter_paged_mode(moreStrPage);
2148
2149 while (argN) {
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002150 WCHAR *thisArg = WCMD_parameter (command, argno++, &argN);
Jason Edmeadesce875222007-04-13 21:34:05 +01002151 HANDLE h;
2152
2153 if (!argN) break;
2154
2155 if (needsPause) {
2156
2157 /* Wait */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002158 wsprintf(moreStrPage, moreFmt2, moreStr, 100);
Jason Edmeadesce875222007-04-13 21:34:05 +01002159 WCMD_leave_paged_mode();
2160 WCMD_output_asis(moreStrPage);
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002161 WCMD_ReadFile (GetStdHandle(STD_INPUT_HANDLE), buffer,
2162 sizeof(buffer)/sizeof(WCHAR), &count, NULL);
Jason Edmeadesce875222007-04-13 21:34:05 +01002163 WCMD_enter_paged_mode(moreStrPage);
2164 }
2165
2166
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002167 WINE_TRACE("more: Processing arg '%s'\n", wine_dbgstr_w(thisArg));
Jason Edmeadesce875222007-04-13 21:34:05 +01002168 h = CreateFile (thisArg, GENERIC_READ, FILE_SHARE_READ, NULL, OPEN_EXISTING,
2169 FILE_ATTRIBUTE_NORMAL, NULL);
2170 if (h == INVALID_HANDLE_VALUE) {
2171 WCMD_print_error ();
Jason Edmeades5cc492c2007-06-03 22:07:39 +01002172 WCMD_output (WCMD_LoadMessage(WCMD_READFAIL), thisArg);
Jason Edmeadesce875222007-04-13 21:34:05 +01002173 errorlevel = 1;
2174 } else {
2175 ULONG64 curPos = 0;
2176 ULONG64 fileLen = 0;
2177 WIN32_FILE_ATTRIBUTE_DATA fileInfo;
2178
2179 /* Get the file size */
2180 GetFileAttributesEx(thisArg, GetFileExInfoStandard, (void*)&fileInfo);
2181 fileLen = (((ULONG64)fileInfo.nFileSizeHigh) << 32) + fileInfo.nFileSizeLow;
2182
2183 needsPause = TRUE;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002184 while (WCMD_ReadFile (h, buffer, (sizeof(buffer)/sizeof(WCHAR))-1, &count, NULL)) {
Jason Edmeadesce875222007-04-13 21:34:05 +01002185 if (count == 0) break; /* ReadFile reports success on EOF! */
2186 buffer[count] = 0;
2187 curPos += count;
2188
2189 /* Update % count (would be used in WCMD_output_asis as prompt) */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002190 wsprintf(moreStrPage, moreFmt2, moreStr, (int) min(99, (curPos * 100)/fileLen));
Jason Edmeadesce875222007-04-13 21:34:05 +01002191
2192 WCMD_output_asis (buffer);
2193 }
2194 CloseHandle (h);
2195 }
2196 }
2197
2198 WCMD_leave_paged_mode();
2199 }
2200}
2201
2202/****************************************************************************
Dave Pickles74f440e1999-06-06 15:24:04 +00002203 * WCMD_verify
2204 *
2205 * Display verify flag.
Dave Pickles5f8f4f71999-06-26 10:24:08 +00002206 * FIXME: We don't actually do anything with the verify flag other than toggle
2207 * it...
Dave Pickles74f440e1999-06-06 15:24:04 +00002208 */
2209
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002210void WCMD_verify (WCHAR *command) {
Dave Pickles74f440e1999-06-06 15:24:04 +00002211
Alexandre Julliardabfe1052007-03-01 12:43:19 +01002212 int count;
Dave Pickles74f440e1999-06-06 15:24:04 +00002213
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002214 count = strlenW(command);
Dave Pickles5f8f4f71999-06-26 10:24:08 +00002215 if (count == 0) {
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002216 if (verify_mode) WCMD_output (WCMD_LoadMessage(WCMD_VERIFYPROMPT), onW);
2217 else WCMD_output (WCMD_LoadMessage(WCMD_VERIFYPROMPT), offW);
Dave Pickles5f8f4f71999-06-26 10:24:08 +00002218 return;
2219 }
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002220 if (lstrcmpiW(command, onW) == 0) {
Dave Pickles5f8f4f71999-06-26 10:24:08 +00002221 verify_mode = 1;
2222 return;
2223 }
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002224 else if (lstrcmpiW(command, offW) == 0) {
Dave Pickles5f8f4f71999-06-26 10:24:08 +00002225 verify_mode = 0;
2226 return;
2227 }
Jason Edmeades5cc492c2007-06-03 22:07:39 +01002228 else WCMD_output (WCMD_LoadMessage(WCMD_VERIFYERR));
Dave Pickles74f440e1999-06-06 15:24:04 +00002229}
2230
2231/****************************************************************************
2232 * WCMD_version
2233 *
2234 * Display version info.
2235 */
2236
Eric Pouechb2f079b2003-02-27 01:41:21 +00002237void WCMD_version (void) {
Dave Pickles74f440e1999-06-06 15:24:04 +00002238
2239 WCMD_output (version_string);
2240
2241}
2242
2243/****************************************************************************
2244 * WCMD_volume
2245 *
2246 * Display volume info and/or set volume label. Returns 0 if error.
2247 */
2248
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002249int WCMD_volume (int mode, WCHAR *path) {
Dave Pickles74f440e1999-06-06 15:24:04 +00002250
Alexandre Julliardabfe1052007-03-01 12:43:19 +01002251 DWORD count, serial;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002252 WCHAR string[MAX_PATH], label[MAX_PATH], curdir[MAX_PATH];
Alexandre Julliardabfe1052007-03-01 12:43:19 +01002253 BOOL status;
Dave Pickles74f440e1999-06-06 15:24:04 +00002254
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002255 if (strlenW(path) == 0) {
2256 status = GetCurrentDirectory (sizeof(curdir)/sizeof(WCHAR), curdir);
Dave Pickles74f440e1999-06-06 15:24:04 +00002257 if (!status) {
2258 WCMD_print_error ();
2259 return 0;
2260 }
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002261 status = GetVolumeInformation (NULL, label, sizeof(label)/sizeof(WCHAR),
2262 &serial, NULL, NULL, NULL, 0);
Dave Pickles74f440e1999-06-06 15:24:04 +00002263 }
2264 else {
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002265 static const WCHAR fmt[] = {'%','s','\\','\0'};
2266 if ((path[1] != ':') || (strlenW(path) != 2)) {
Jason Edmeades5cc492c2007-06-03 22:07:39 +01002267 WCMD_output (WCMD_LoadMessage(WCMD_SYNTAXERR));
Dave Pickles74f440e1999-06-06 15:24:04 +00002268 return 0;
2269 }
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002270 wsprintf (curdir, fmt, path);
2271 status = GetVolumeInformation (curdir, label, sizeof(label)/sizeof(WCHAR),
2272 &serial, NULL,
Dave Pickles74f440e1999-06-06 15:24:04 +00002273 NULL, NULL, 0);
2274 }
2275 if (!status) {
2276 WCMD_print_error ();
2277 return 0;
2278 }
Jason Edmeades5cc492c2007-06-03 22:07:39 +01002279 WCMD_output (WCMD_LoadMessage(WCMD_VOLUMEDETAIL),
Dave Pickles74f440e1999-06-06 15:24:04 +00002280 curdir[0], label, HIWORD(serial), LOWORD(serial));
2281 if (mode) {
Jason Edmeades5cc492c2007-06-03 22:07:39 +01002282 WCMD_output (WCMD_LoadMessage(WCMD_VOLUMEPROMPT));
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002283 WCMD_ReadFile (GetStdHandle(STD_INPUT_HANDLE), string,
2284 sizeof(string)/sizeof(WCHAR), &count, NULL);
Dave Pickles036a9f71999-07-10 11:36:45 +00002285 if (count > 1) {
2286 string[count-1] = '\0'; /* ReadFile output is not null-terminated! */
2287 if (string[count-2] == '\r') string[count-2] = '\0'; /* Under Windoze we get CRLF! */
2288 }
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002289 if (strlenW(path) != 0) {
Dave Pickles74f440e1999-06-06 15:24:04 +00002290 if (!SetVolumeLabel (curdir, string)) WCMD_print_error ();
2291 }
2292 else {
2293 if (!SetVolumeLabel (NULL, string)) WCMD_print_error ();
2294 }
2295 }
2296 return 1;
2297}
Jason Edmeadesc3666482007-02-20 17:49:08 +00002298
2299/**************************************************************************
2300 * WCMD_exit
2301 *
2302 * Exit either the process, or just this batch program
2303 *
2304 */
2305
Jason Edmeadesd2474de2007-06-15 20:59:24 +01002306void WCMD_exit (CMD_LIST **cmdList) {
Jason Edmeadesc3666482007-02-20 17:49:08 +00002307
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002308 static const WCHAR parmB[] = {'/','B','\0'};
2309 int rc = atoiW(param1); /* Note: atoi of empty parameter is 0 */
Jason Edmeadesc3666482007-02-20 17:49:08 +00002310
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002311 if (context && lstrcmpiW(quals, parmB) == 0) {
Jason Edmeadesc3666482007-02-20 17:49:08 +00002312 errorlevel = rc;
2313 context -> skip_rest = TRUE;
Jason Edmeadesd2474de2007-06-15 20:59:24 +01002314 *cmdList = NULL;
Jason Edmeadesc3666482007-02-20 17:49:08 +00002315 } else {
2316 ExitProcess(rc);
2317 }
2318}
Jason Edmeadesfda72292007-02-27 23:18:57 +00002319
2320/**************************************************************************
2321 * WCMD_ask_confirm
2322 *
2323 * Issue a message and ask 'Are you sure (Y/N)', waiting on a valid
2324 * answer.
2325 *
Jason Edmeades5cc492c2007-06-03 22:07:39 +01002326 * Returns True if Y (or A) answer is selected
2327 * If optionAll contains a pointer, ALL is allowed, and if answered
2328 * set to TRUE
Jason Edmeadesfda72292007-02-27 23:18:57 +00002329 *
2330 */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002331BOOL WCMD_ask_confirm (WCHAR *message, BOOL showSureText, BOOL *optionAll) {
Jason Edmeadesfda72292007-02-27 23:18:57 +00002332
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002333 WCHAR msgbuffer[MAXSTRING];
2334 WCHAR Ybuffer[MAXSTRING];
2335 WCHAR Nbuffer[MAXSTRING];
2336 WCHAR Abuffer[MAXSTRING];
2337 WCHAR answer[MAX_PATH] = {'\0'};
Jason Edmeadesfda72292007-02-27 23:18:57 +00002338 DWORD count = 0;
2339
2340 /* Load the translated 'Are you sure', plus valid answers */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002341 LoadString (hinst, WCMD_CONFIRM, msgbuffer, sizeof(msgbuffer)/sizeof(WCHAR));
2342 LoadString (hinst, WCMD_YES, Ybuffer, sizeof(Ybuffer)/sizeof(WCHAR));
2343 LoadString (hinst, WCMD_NO, Nbuffer, sizeof(Nbuffer)/sizeof(WCHAR));
2344 LoadString (hinst, WCMD_ALL, Abuffer, sizeof(Abuffer)/sizeof(WCHAR));
Jason Edmeadesfda72292007-02-27 23:18:57 +00002345
2346 /* Loop waiting on a Y or N */
2347 while (answer[0] != Ybuffer[0] && answer[0] != Nbuffer[0]) {
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002348 static const WCHAR startBkt[] = {' ','(','\0'};
2349 static const WCHAR endBkt[] = {')','?','\0'};
2350
Jason Edmeadesfda72292007-02-27 23:18:57 +00002351 WCMD_output_asis (message);
Jason Edmeadesfe29ed42007-02-27 23:20:05 +00002352 if (showSureText) {
2353 WCMD_output_asis (msgbuffer);
2354 }
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002355 WCMD_output_asis (startBkt);
Jason Edmeadesfda72292007-02-27 23:18:57 +00002356 WCMD_output_asis (Ybuffer);
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002357 WCMD_output_asis (fslashW);
Jason Edmeadesfda72292007-02-27 23:18:57 +00002358 WCMD_output_asis (Nbuffer);
Jason Edmeades5cc492c2007-06-03 22:07:39 +01002359 if (optionAll) {
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002360 WCMD_output_asis (fslashW);
Jason Edmeades5cc492c2007-06-03 22:07:39 +01002361 WCMD_output_asis (Abuffer);
2362 }
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002363 WCMD_output_asis (endBkt);
2364 WCMD_ReadFile (GetStdHandle(STD_INPUT_HANDLE), answer,
2365 sizeof(answer)/sizeof(WCHAR), &count, NULL);
Anatoly Lyutin24866e72007-07-09 12:27:50 +04002366 answer[0] = toupperW(answer[0]);
Jason Edmeadesfda72292007-02-27 23:18:57 +00002367 }
2368
2369 /* Return the answer */
Jason Edmeades5cc492c2007-06-03 22:07:39 +01002370 return ((answer[0] == Ybuffer[0]) ||
2371 (optionAll && (answer[0] == Abuffer[0])));
Jason Edmeadesfda72292007-02-27 23:18:57 +00002372}
Jason Edmeades79aa1a02007-03-04 22:34:45 +00002373
2374/*****************************************************************************
2375 * WCMD_assoc
2376 *
Jason Edmeades9e041c62007-03-13 00:06:58 +00002377 * Lists or sets file associations (assoc = TRUE)
2378 * Lists or sets file types (assoc = FALSE)
Jason Edmeades79aa1a02007-03-04 22:34:45 +00002379 */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002380void WCMD_assoc (WCHAR *command, BOOL assoc) {
Jason Edmeades79aa1a02007-03-04 22:34:45 +00002381
2382 HKEY key;
2383 DWORD accessOptions = KEY_READ;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002384 WCHAR *newValue;
Jason Edmeades79aa1a02007-03-04 22:34:45 +00002385 LONG rc = ERROR_SUCCESS;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002386 WCHAR keyValue[MAXSTRING];
Jason Edmeades79aa1a02007-03-04 22:34:45 +00002387 DWORD valueLen = MAXSTRING;
2388 HKEY readKey;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002389 static const WCHAR shOpCmdW[] = {'\\','S','h','e','l','l','\\',
2390 'O','p','e','n','\\','C','o','m','m','a','n','d','\0'};
Jason Edmeades79aa1a02007-03-04 22:34:45 +00002391
2392 /* See if parameter includes '=' */
2393 errorlevel = 0;
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002394 newValue = strchrW(command, '=');
Jason Edmeades79aa1a02007-03-04 22:34:45 +00002395 if (newValue) accessOptions |= KEY_WRITE;
2396
2397 /* Open a key to HKEY_CLASSES_ROOT for enumerating */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002398 if (RegOpenKeyEx(HKEY_CLASSES_ROOT, nullW, 0,
Jason Edmeades79aa1a02007-03-04 22:34:45 +00002399 accessOptions, &key) != ERROR_SUCCESS) {
2400 WINE_FIXME("Unexpected failure opening HKCR key: %d\n", GetLastError());
2401 return;
2402 }
2403
Francois Gouget7b0cde82007-03-06 14:26:24 +01002404 /* If no parameters then list all associations */
Jason Edmeades79aa1a02007-03-04 22:34:45 +00002405 if (*command == 0x00) {
2406 int index = 0;
2407
2408 /* Enumerate all the keys */
2409 while (rc != ERROR_NO_MORE_ITEMS) {
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002410 WCHAR keyName[MAXSTRING];
Jason Edmeades79aa1a02007-03-04 22:34:45 +00002411 DWORD nameLen;
2412
2413 /* Find the next value */
2414 nameLen = MAXSTRING;
2415 rc = RegEnumKeyEx(key, index++,
2416 keyName, &nameLen,
2417 NULL, NULL, NULL, NULL);
2418
2419 if (rc == ERROR_SUCCESS) {
2420
Jason Edmeades9e041c62007-03-13 00:06:58 +00002421 /* Only interested in extension ones if assoc, or others
2422 if not assoc */
2423 if ((keyName[0] == '.' && assoc) ||
2424 (!(keyName[0] == '.') && (!assoc)))
2425 {
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002426 WCHAR subkey[MAXSTRING];
2427 strcpyW(subkey, keyName);
2428 if (!assoc) strcatW(subkey, shOpCmdW);
Jason Edmeades79aa1a02007-03-04 22:34:45 +00002429
Jason Edmeades9e041c62007-03-13 00:06:58 +00002430 if (RegOpenKeyEx(key, subkey, 0,
Jason Edmeades79aa1a02007-03-04 22:34:45 +00002431 accessOptions, &readKey) == ERROR_SUCCESS) {
2432
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002433 valueLen = sizeof(keyValue)/sizeof(WCHAR);
Jason Edmeades79aa1a02007-03-04 22:34:45 +00002434 rc = RegQueryValueEx(readKey, NULL, NULL, NULL,
2435 (LPBYTE)keyValue, &valueLen);
2436 WCMD_output_asis(keyName);
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002437 WCMD_output_asis(equalW);
Jason Edmeades79aa1a02007-03-04 22:34:45 +00002438 /* If no default value found, leave line empty after '=' */
2439 if (rc == ERROR_SUCCESS) {
2440 WCMD_output_asis(keyValue);
2441 }
Jason Edmeades5cc492c2007-06-03 22:07:39 +01002442 WCMD_output_asis(newline);
Jason Edmeades79aa1a02007-03-04 22:34:45 +00002443 }
2444 }
2445 }
2446 }
2447 RegCloseKey(readKey);
2448
2449 } else {
2450
2451 /* Parameter supplied - if no '=' on command line, its a query */
2452 if (newValue == NULL) {
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002453 WCHAR *space;
2454 WCHAR subkey[MAXSTRING];
Jason Edmeades79aa1a02007-03-04 22:34:45 +00002455
2456 /* Query terminates the parameter at the first space */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002457 strcpyW(keyValue, command);
2458 space = strchrW(keyValue, ' ');
Jason Edmeades79aa1a02007-03-04 22:34:45 +00002459 if (space) *space=0x00;
2460
Jason Edmeades9e041c62007-03-13 00:06:58 +00002461 /* Set up key name */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002462 strcpyW(subkey, keyValue);
2463 if (!assoc) strcatW(subkey, shOpCmdW);
Jason Edmeades9e041c62007-03-13 00:06:58 +00002464
2465 if (RegOpenKeyEx(key, subkey, 0,
Jason Edmeades79aa1a02007-03-04 22:34:45 +00002466 accessOptions, &readKey) == ERROR_SUCCESS) {
2467
2468 rc = RegQueryValueEx(readKey, NULL, NULL, NULL,
2469 (LPBYTE)keyValue, &valueLen);
2470 WCMD_output_asis(command);
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002471 WCMD_output_asis(equalW);
Jason Edmeades79aa1a02007-03-04 22:34:45 +00002472 /* If no default value found, leave line empty after '=' */
2473 if (rc == ERROR_SUCCESS) WCMD_output_asis(keyValue);
Jason Edmeades5cc492c2007-06-03 22:07:39 +01002474 WCMD_output_asis(newline);
Jason Edmeades79aa1a02007-03-04 22:34:45 +00002475 RegCloseKey(readKey);
2476
2477 } else {
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002478 WCHAR msgbuffer[MAXSTRING];
2479 WCHAR outbuffer[MAXSTRING];
Jason Edmeades79aa1a02007-03-04 22:34:45 +00002480
2481 /* Load the translated 'File association not found' */
Jason Edmeades9e041c62007-03-13 00:06:58 +00002482 if (assoc) {
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002483 LoadString (hinst, WCMD_NOASSOC, msgbuffer, sizeof(msgbuffer)/sizeof(WCHAR));
Jason Edmeades9e041c62007-03-13 00:06:58 +00002484 } else {
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002485 LoadString (hinst, WCMD_NOFTYPE, msgbuffer, sizeof(msgbuffer)/sizeof(WCHAR));
Jason Edmeades9e041c62007-03-13 00:06:58 +00002486 }
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002487 wsprintf(outbuffer, msgbuffer, keyValue);
Jason Edmeades79aa1a02007-03-04 22:34:45 +00002488 WCMD_output_asis(outbuffer);
2489 errorlevel = 2;
2490 }
2491
2492 /* Not a query - its a set or clear of a value */
2493 } else {
2494
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002495 WCHAR subkey[MAXSTRING];
Jason Edmeades9e041c62007-03-13 00:06:58 +00002496
Jason Edmeades79aa1a02007-03-04 22:34:45 +00002497 /* Get pointer to new value */
2498 *newValue = 0x00;
2499 newValue++;
2500
Jason Edmeades9e041c62007-03-13 00:06:58 +00002501 /* Set up key name */
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002502 strcpyW(subkey, command);
2503 if (!assoc) strcatW(subkey, shOpCmdW);
Jason Edmeades9e041c62007-03-13 00:06:58 +00002504
2505 /* If nothing after '=' then clear value - only valid for ASSOC */
Jason Edmeades79aa1a02007-03-04 22:34:45 +00002506 if (*newValue == 0x00) {
2507
Jason Edmeades9e041c62007-03-13 00:06:58 +00002508 if (assoc) rc = RegDeleteKey(key, command);
2509 if (assoc && rc == ERROR_SUCCESS) {
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002510 WINE_TRACE("HKCR Key '%s' deleted\n", wine_dbgstr_w(command));
Jason Edmeades79aa1a02007-03-04 22:34:45 +00002511
Jason Edmeades9e041c62007-03-13 00:06:58 +00002512 } else if (assoc && rc != ERROR_FILE_NOT_FOUND) {
Jason Edmeades79aa1a02007-03-04 22:34:45 +00002513 WCMD_print_error();
2514 errorlevel = 2;
2515
2516 } else {
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002517 WCHAR msgbuffer[MAXSTRING];
2518 WCHAR outbuffer[MAXSTRING];
Jason Edmeades79aa1a02007-03-04 22:34:45 +00002519
2520 /* Load the translated 'File association not found' */
Jason Edmeades9e041c62007-03-13 00:06:58 +00002521 if (assoc) {
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002522 LoadString (hinst, WCMD_NOASSOC, msgbuffer,
2523 sizeof(msgbuffer)/sizeof(WCHAR));
Jason Edmeades9e041c62007-03-13 00:06:58 +00002524 } else {
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002525 LoadString (hinst, WCMD_NOFTYPE, msgbuffer,
2526 sizeof(msgbuffer)/sizeof(WCHAR));
Jason Edmeades9e041c62007-03-13 00:06:58 +00002527 }
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002528 wsprintf(outbuffer, msgbuffer, keyValue);
Jason Edmeades79aa1a02007-03-04 22:34:45 +00002529 WCMD_output_asis(outbuffer);
2530 errorlevel = 2;
2531 }
2532
2533 /* It really is a set value = contents */
2534 } else {
Jason Edmeades9e041c62007-03-13 00:06:58 +00002535 rc = RegCreateKeyEx(key, subkey, 0, NULL, REG_OPTION_NON_VOLATILE,
Jason Edmeades79aa1a02007-03-04 22:34:45 +00002536 accessOptions, NULL, &readKey, NULL);
2537 if (rc == ERROR_SUCCESS) {
2538 rc = RegSetValueEx(readKey, NULL, 0, REG_SZ,
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002539 (LPBYTE)newValue, strlenW(newValue));
Jason Edmeades79aa1a02007-03-04 22:34:45 +00002540 RegCloseKey(readKey);
2541 }
2542
2543 if (rc != ERROR_SUCCESS) {
2544 WCMD_print_error();
2545 errorlevel = 2;
2546 } else {
2547 WCMD_output_asis(command);
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002548 WCMD_output_asis(equalW);
Jason Edmeades79aa1a02007-03-04 22:34:45 +00002549 WCMD_output_asis(newValue);
Jason Edmeades5cc492c2007-06-03 22:07:39 +01002550 WCMD_output_asis(newline);
Jason Edmeades79aa1a02007-03-04 22:34:45 +00002551 }
2552 }
2553 }
2554 }
2555
2556 /* Clean up */
2557 RegCloseKey(key);
2558}
Jason Edmeadese37463f2007-03-08 00:37:44 +00002559
2560/****************************************************************************
2561 * WCMD_color
2562 *
2563 * Clear the terminal screen.
2564 */
2565
2566void WCMD_color (void) {
2567
2568 /* Emulate by filling the screen from the top left to bottom right with
2569 spaces, then moving the cursor to the top left afterwards */
2570 CONSOLE_SCREEN_BUFFER_INFO consoleInfo;
2571 HANDLE hStdOut = GetStdHandle(STD_OUTPUT_HANDLE);
2572
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002573 if (param1[0] != 0x00 && strlenW(param1) > 2) {
Jason Edmeades5cc492c2007-06-03 22:07:39 +01002574 WCMD_output (WCMD_LoadMessage(WCMD_ARGERR));
Jason Edmeadese37463f2007-03-08 00:37:44 +00002575 return;
2576 }
2577
2578 if (GetConsoleScreenBufferInfo(hStdOut, &consoleInfo))
2579 {
2580 COORD topLeft;
2581 DWORD screenSize;
2582 DWORD color = 0;
2583
2584 screenSize = consoleInfo.dwSize.X * (consoleInfo.dwSize.Y + 1);
2585
2586 topLeft.X = 0;
2587 topLeft.Y = 0;
2588
2589 /* Convert the color hex digits */
2590 if (param1[0] == 0x00) {
2591 color = defaultColor;
2592 } else {
Jason Edmeadesb8aa5fc2007-06-03 22:07:42 +01002593 color = strtoulW(param1, NULL, 16);
Jason Edmeadese37463f2007-03-08 00:37:44 +00002594 }
2595
2596 /* Fail if fg == bg color */
2597 if (((color & 0xF0) >> 4) == (color & 0x0F)) {
2598 errorlevel = 1;
2599 return;
2600 }
2601
2602 /* Set the current screen contents and ensure all future writes
2603 remain this color */
2604 FillConsoleOutputAttribute(hStdOut, color, screenSize, topLeft, &screenSize);
2605 SetConsoleTextAttribute(hStdOut, color);
2606 }
2607}