2 * Copyright 2009 Dan Kegel
3 * Copyright 2010 Jacek Caban for CodeWeavers
5 * This library is free software; you can redistribute it and/or
6 * modify it under the terms of the GNU Lesser General Public
7 * License as published by the Free Software Foundation; either
8 * version 2.1 of the License, or (at your option) any later version.
10 * This library is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 * Lesser General Public License for more details.
15 * You should have received a copy of the GNU Lesser General Public
16 * License along with this library; if not, write to the Free Software
17 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
23 #include "wine/test.h"
25 static char workdir[MAX_PATH];
26 static DWORD workdir_len;
28 /* Substitute escaped spaces with real ones */
29 static const char* replace_escaped_spaces(const char *data, DWORD size, DWORD *new_size)
31 static const char escaped_space[] = {'@','s','p','a','c','e','@','\0'};
34 DWORD len_space = sizeof(escaped_space) -1;
39 new_data = HeapAlloc(GetProcessHeap(), 0, size*sizeof(char));
40 ok(new_data != NULL, "HeapAlloc failed\n");
44 while( (b = strstr(a, escaped_space)) )
46 strncpy(new_data + *new_size, a, b-a + 1);
48 new_data[*new_size - 1] = ' ';
52 strncpy(new_data + *new_size, a, strlen(a) + 1);
53 *new_size += strlen(a);
58 static BOOL run_cmd(const char *cmd_data, DWORD cmd_size)
60 SECURITY_ATTRIBUTES sa = {sizeof(sa), 0, TRUE};
61 char command[] = "test.cmd";
62 STARTUPINFOA si = {sizeof(si)};
63 PROCESS_INFORMATION pi;
68 file = CreateFileA("test.cmd", GENERIC_WRITE, 0, NULL, CREATE_ALWAYS,
69 FILE_ATTRIBUTE_NORMAL, NULL);
70 ok(file != INVALID_HANDLE_VALUE, "CreateFile failed\n");
71 if(file == INVALID_HANDLE_VALUE)
74 bres = WriteFile(file, cmd_data, cmd_size, &size, NULL);
76 ok(bres, "Could not write to file: %u\n", GetLastError());
80 file = CreateFileA("test.out", GENERIC_WRITE, FILE_SHARE_WRITE|FILE_SHARE_READ, &sa, CREATE_ALWAYS,
81 FILE_ATTRIBUTE_NORMAL, NULL);
82 ok(file != INVALID_HANDLE_VALUE, "CreateFile failed\n");
83 if(file == INVALID_HANDLE_VALUE)
86 fileerr = CreateFileA("test.err", GENERIC_WRITE, FILE_SHARE_WRITE|FILE_SHARE_READ, &sa, CREATE_ALWAYS,
87 FILE_ATTRIBUTE_NORMAL, NULL);
88 ok(fileerr != INVALID_HANDLE_VALUE, "CreateFile stderr failed\n");
89 if(fileerr == INVALID_HANDLE_VALUE)
92 si.dwFlags = STARTF_USESTDHANDLES;
94 si.hStdError = fileerr;
95 bres = CreateProcessA(NULL, command, NULL, NULL, TRUE, 0, NULL, NULL, &si, &pi);
96 ok(bres, "CreateProcess failed: %u\n", GetLastError());
98 DeleteFileA("test.out");
102 WaitForSingleObject(pi.hProcess, INFINITE);
103 CloseHandle(pi.hThread);
104 CloseHandle(pi.hProcess);
106 CloseHandle(fileerr);
107 DeleteFileA("test.cmd");
111 static DWORD map_file(const char *file_name, const char **ret)
116 file = CreateFileA(file_name, GENERIC_READ, 0, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_READONLY, NULL);
117 ok(file != INVALID_HANDLE_VALUE, "CreateFile failed: %08x\n", GetLastError());
118 if(file == INVALID_HANDLE_VALUE)
121 size = GetFileSize(file, NULL);
123 map = CreateFileMapping(file, NULL, PAGE_READONLY, 0, 0, NULL);
125 ok(map != NULL, "CreateFileMapping(%s) failed: %u\n", file_name, GetLastError());
129 *ret = MapViewOfFile(map, FILE_MAP_READ, 0, 0, 0);
130 ok(*ret != NULL, "MapViewOfFile failed: %u\n", GetLastError());
138 static const char *compare_line(const char *out_line, const char *out_end, const char *exp_line,
141 const char *out_ptr = out_line, *exp_ptr = exp_line;
142 const char *err = NULL;
144 static const char pwd_cmd[] = {'@','p','w','d','@'};
145 static const char todo_space_cmd[] = {'@','t','o','d','o','_','s','p','a','c','e','@'};
146 static const char space_cmd[] = {'@','s','p','a','c','e','@'};
147 static const char or_broken_cmd[] = {'@','o','r','_','b','r','o','k','e','n','@'};
149 while(exp_ptr < exp_end) {
150 if(*exp_ptr == '@') {
151 if(exp_ptr+sizeof(pwd_cmd) <= exp_end
152 && !memcmp(exp_ptr, pwd_cmd, sizeof(pwd_cmd))) {
153 exp_ptr += sizeof(pwd_cmd);
154 if(out_end-out_ptr < workdir_len
155 || (CompareStringA(LOCALE_SYSTEM_DEFAULT, NORM_IGNORECASE, out_ptr, workdir_len,
156 workdir, workdir_len) != CSTR_EQUAL)) {
159 out_ptr += workdir_len;
162 }else if(exp_ptr+sizeof(todo_space_cmd) <= exp_end
163 && !memcmp(exp_ptr, todo_space_cmd, sizeof(todo_space_cmd))) {
164 exp_ptr += sizeof(todo_space_cmd);
165 todo_wine ok(*out_ptr == ' ', "expected space\n");
166 if(out_ptr < out_end && *out_ptr == ' ')
169 }else if(exp_ptr+sizeof(space_cmd) <= exp_end
170 && !memcmp(exp_ptr, space_cmd, sizeof(space_cmd))) {
171 exp_ptr += sizeof(space_cmd);
172 ok(*out_ptr == ' ', "expected space\n");
173 if(out_ptr < out_end && *out_ptr == ' ')
176 }else if(exp_ptr+sizeof(or_broken_cmd) <= exp_end
177 && !memcmp(exp_ptr, or_broken_cmd, sizeof(or_broken_cmd))) {
181 }else if(out_ptr == out_end || *out_ptr != *exp_ptr) {
189 while(exp_ptr+sizeof(or_broken_cmd) <= exp_end && memcmp(exp_ptr, or_broken_cmd, sizeof(or_broken_cmd)))
194 exp_ptr += sizeof(or_broken_cmd);
204 return exp_ptr == exp_end ? NULL : out_ptr;
207 static void test_output(const char *out_data, DWORD out_size, const char *exp_data, DWORD exp_size)
209 const char *out_ptr = out_data, *exp_ptr = exp_data, *out_nl, *exp_nl, *err;
212 while(out_ptr < out_data+out_size && exp_ptr < exp_data+exp_size) {
215 for(exp_nl = exp_ptr; exp_nl < exp_data+exp_size && *exp_nl != '\r' && *exp_nl != '\n'; exp_nl++);
216 for(out_nl = out_ptr; out_nl < out_data+out_size && *out_nl != '\r' && *out_nl != '\n'; out_nl++);
218 err = compare_line(out_ptr, out_nl, exp_ptr, exp_nl);
220 ok(0, "unexpected end of line %d (got '%.*s', wanted '%.*s')\n",
221 line, (int)(out_nl-out_ptr), out_ptr, (int)(exp_nl-exp_ptr), exp_ptr);
223 ok(0, "unexpected char 0x%x position %d in line %d (got '%.*s', wanted '%.*s')\n",
224 *err, (int)(err-out_ptr), line, (int)(out_nl-out_ptr), out_ptr, (int)(exp_nl-exp_ptr), exp_ptr);
228 if(out_nl+1 < out_data+out_size && out_nl[0] == '\r' && out_nl[1] == '\n')
230 if(exp_nl+1 < exp_data+exp_size && exp_nl[0] == '\r' && exp_nl[1] == '\n')
234 ok(exp_ptr >= exp_data+exp_size, "unexpected end of output in line %d, missing %s\n", line, exp_ptr);
235 ok(out_ptr >= out_data+out_size, "too long output, got additional %s\n", out_ptr);
238 static void run_test(const char *cmd_data, DWORD cmd_size, const char *exp_data, DWORD exp_size)
240 const char *out_data, *actual_cmd_data;
241 DWORD out_size, actual_cmd_size;
243 actual_cmd_data = replace_escaped_spaces(cmd_data, cmd_size, &actual_cmd_size);
244 if(!actual_cmd_size || !actual_cmd_data)
247 if(!run_cmd(actual_cmd_data, actual_cmd_size))
250 out_size = map_file("test.out", &out_data);
252 test_output(out_data, out_size, exp_data, exp_size);
253 UnmapViewOfFile(out_data);
255 DeleteFileA("test.out");
256 DeleteFileA("test.err");
259 HeapFree(GetProcessHeap(), 0, (LPVOID)actual_cmd_data);
262 static void run_from_file(char *file_name)
264 char out_name[MAX_PATH];
265 const char *test_data, *out_data;
266 DWORD test_size, out_size;
268 test_size = map_file(file_name, &test_data);
270 ok(0, "Could not map file %s: %u\n", file_name, GetLastError());
274 sprintf(out_name, "%s.exp", file_name);
275 out_size = map_file(out_name, &out_data);
277 ok(0, "Could not map file %s: %u\n", out_name, GetLastError());
278 UnmapViewOfFile(test_data);
282 run_test(test_data, test_size, out_data, out_size);
284 UnmapViewOfFile(test_data);
285 UnmapViewOfFile(out_data);
288 static DWORD load_resource(const char *name, const char *type, const char **ret)
294 src = FindResourceA(NULL, name, type);
295 ok(src != NULL, "Could not find resource %s: %u\n", name, GetLastError());
299 res = LoadResource(NULL, src);
300 size = SizeofResource(NULL, src);
301 while(size && !res[size-1])
308 static BOOL WINAPI test_enum_proc(HMODULE module, LPCTSTR type, LPSTR name, LONG_PTR param)
310 const char *cmd_data, *out_data;
311 DWORD cmd_size, out_size;
314 trace("running %s test...\n", name);
316 cmd_size = load_resource(name, type, &cmd_data);
320 sprintf(res_name, "%s.exp", name);
321 out_size = load_resource(res_name, "TESTOUT", &out_data);
325 run_test(cmd_data, cmd_size, out_data, out_size);
329 static int cmd_available(void)
332 PROCESS_INFORMATION pi;
333 char cmd[] = "cmd /c exit 0";
335 memset(&si, 0, sizeof(si));
337 if (CreateProcessA(NULL, cmd, NULL, NULL, FALSE, 0, NULL, NULL, &si, &pi)) {
338 CloseHandle(pi.hThread);
339 CloseHandle(pi.hProcess);
350 if (!cmd_available()) {
351 win_skip("cmd not installed, skipping cmd tests\n");
355 workdir_len = GetCurrentDirectoryA(sizeof(workdir), workdir);
357 argc = winetest_get_mainargs(&argv);
359 run_from_file(argv[2]);
361 EnumResourceNamesA(NULL, "TESTCMD", test_enum_proc, 0);