Reactos
at listview 492 lines 17 kB view raw
1/* 2 * Copyright 2009 Dan Kegel 3 * Copyright 2010 Jacek Caban for CodeWeavers 4 * 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. 9 * 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. 14 * 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 18 */ 19 20//#include <windows.h> 21#include <stdio.h> 22 23#include <wine/test.h> 24#include <winnls.h> 25 26static char workdir[MAX_PATH]; 27static DWORD workdir_len; 28static char drive[2]; 29static const DWORD drive_len = ARRAY_SIZE(drive); 30static char path[MAX_PATH]; 31static DWORD path_len; 32static char shortpath[MAX_PATH]; 33static DWORD shortpath_len; 34 35/* Convert to DOS line endings, and substitute escaped whitespace chars with real ones */ 36static const char* convert_input_data(const char *data, DWORD size, DWORD *new_size) 37{ 38 static const char escaped_space[] = {'@','s','p','a','c','e','@'}; 39 static const char escaped_tab[] = {'@','t','a','b','@'}; 40 DWORD i, eol_count = 0; 41 char *ptr, *new_data; 42 43 for (i = 0; i < size; i++) 44 if (data[i] == '\n') eol_count++; 45 46 ptr = new_data = HeapAlloc(GetProcessHeap(), 0, size + eol_count + 1); 47 48 for (i = 0; i < size; i++) { 49 switch (data[i]) { 50 case '\n': 51 if (data[i-1] != '\r') 52 *ptr++ = '\r'; 53 *ptr++ = '\n'; 54 break; 55 case '@': 56 if (data + i + sizeof(escaped_space) - 1 < data + size 57 && !memcmp(data + i, escaped_space, sizeof(escaped_space))) { 58 *ptr++ = ' '; 59 i += sizeof(escaped_space) - 1; 60 } else if (data + i + sizeof(escaped_tab) - 1 < data + size 61 && !memcmp(data + i, escaped_tab, sizeof(escaped_tab))) { 62 *ptr++ = '\t'; 63 i += sizeof(escaped_tab) - 1; 64 } else { 65 *ptr++ = data[i]; 66 } 67 break; 68 default: 69 *ptr++ = data[i]; 70 } 71 } 72 *ptr = '\0'; 73 74#ifdef __REACTOS__ 75 *new_size = lstrlenA(new_data); 76#else 77 *new_size = strlen(new_data); 78#endif 79 return new_data; 80} 81 82static BOOL run_cmd(const char *cmd_data, DWORD cmd_size) 83{ 84 SECURITY_ATTRIBUTES sa = {sizeof(sa), 0, TRUE}; 85 char command[] = "test.cmd"; 86 STARTUPINFOA si = {sizeof(si)}; 87 PROCESS_INFORMATION pi; 88 HANDLE file,fileerr; 89 DWORD size; 90 BOOL bres; 91 92 file = CreateFileA("test.cmd", GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, 93 FILE_ATTRIBUTE_NORMAL, NULL); 94 ok(file != INVALID_HANDLE_VALUE, "CreateFile failed\n"); 95 if(file == INVALID_HANDLE_VALUE) 96 return FALSE; 97 98 bres = WriteFile(file, cmd_data, cmd_size, &size, NULL); 99 CloseHandle(file); 100 ok(bres, "Could not write to file: %u\n", GetLastError()); 101 if(!bres) 102 return FALSE; 103 104 file = CreateFileA("test.out", GENERIC_WRITE, FILE_SHARE_WRITE|FILE_SHARE_READ, &sa, CREATE_ALWAYS, 105 FILE_ATTRIBUTE_NORMAL, NULL); 106 ok(file != INVALID_HANDLE_VALUE, "CreateFile failed\n"); 107 if(file == INVALID_HANDLE_VALUE) 108 return FALSE; 109 110 fileerr = CreateFileA("test.err", GENERIC_WRITE, FILE_SHARE_WRITE|FILE_SHARE_READ, &sa, CREATE_ALWAYS, 111 FILE_ATTRIBUTE_NORMAL, NULL); 112 ok(fileerr != INVALID_HANDLE_VALUE, "CreateFile stderr failed\n"); 113 if(fileerr == INVALID_HANDLE_VALUE) 114 return FALSE; 115 116 si.dwFlags = STARTF_USESTDHANDLES; 117 si.hStdOutput = file; 118 si.hStdError = fileerr; 119 bres = CreateProcessA(NULL, command, NULL, NULL, TRUE, 0, NULL, NULL, &si, &pi); 120 ok(bres, "CreateProcess failed: %u\n", GetLastError()); 121 if(!bres) { 122 DeleteFileA("test.out"); 123 return FALSE; 124 } 125 126 WaitForSingleObject(pi.hProcess, INFINITE); 127 CloseHandle(pi.hThread); 128 CloseHandle(pi.hProcess); 129 CloseHandle(file); 130 CloseHandle(fileerr); 131 DeleteFileA("test.cmd"); 132 return TRUE; 133} 134 135static DWORD map_file(const char *file_name, const char **ret) 136{ 137 HANDLE file, map; 138 DWORD size; 139 140 file = CreateFileA(file_name, GENERIC_READ, 0, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_READONLY, NULL); 141 ok(file != INVALID_HANDLE_VALUE, "CreateFile failed: %08x\n", GetLastError()); 142 if(file == INVALID_HANDLE_VALUE) 143 return 0; 144 145 size = GetFileSize(file, NULL); 146 147 map = CreateFileMappingA(file, NULL, PAGE_READONLY, 0, 0, NULL); 148 CloseHandle(file); 149 ok(map != NULL, "CreateFileMappingA(%s) failed: %u\n", file_name, GetLastError()); 150 if(!map) 151 return 0; 152 153 *ret = MapViewOfFile(map, FILE_MAP_READ, 0, 0, 0); 154 ok(*ret != NULL, "MapViewOfFile failed: %u\n", GetLastError()); 155 CloseHandle(map); 156 if(!*ret) 157 return 0; 158 159 return size; 160} 161 162static const char *compare_line(const char *out_line, const char *out_end, const char *exp_line, 163 const char *exp_end) 164{ 165 const char *out_ptr = out_line, *exp_ptr = exp_line; 166 const char *err = NULL; 167 168 static const char pwd_cmd[] = {'@','p','w','d','@'}; 169 static const char drive_cmd[] = {'@','d','r','i','v','e','@'}; 170 static const char path_cmd[] = {'@','p','a','t','h','@'}; 171 static const char shortpath_cmd[] = {'@','s','h','o','r','t','p','a','t','h','@'}; 172 static const char space_cmd[] = {'@','s','p','a','c','e','@'}; 173 static const char spaces_cmd[] = {'@','s','p','a','c','e','s','@'}; 174 static const char tab_cmd[] = {'@','t','a','b','@'}; 175 static const char or_broken_cmd[] = {'@','o','r','_','b','r','o','k','e','n','@'}; 176 177 while(exp_ptr < exp_end) { 178 if(*exp_ptr == '@') { 179 if(exp_ptr+sizeof(pwd_cmd) <= exp_end 180 && !memcmp(exp_ptr, pwd_cmd, sizeof(pwd_cmd))) { 181 exp_ptr += sizeof(pwd_cmd); 182 if(out_end-out_ptr < workdir_len 183 || (CompareStringA(LOCALE_SYSTEM_DEFAULT, NORM_IGNORECASE, out_ptr, workdir_len, 184 workdir, workdir_len) != CSTR_EQUAL)) { 185 err = out_ptr; 186 }else { 187 out_ptr += workdir_len; 188 continue; 189 } 190 } else if(exp_ptr+sizeof(drive_cmd) <= exp_end 191 && !memcmp(exp_ptr, drive_cmd, sizeof(drive_cmd))) { 192 exp_ptr += sizeof(drive_cmd); 193 if(out_end-out_ptr < drive_len 194 || (CompareStringA(LOCALE_SYSTEM_DEFAULT, NORM_IGNORECASE, 195 out_ptr, drive_len, drive, drive_len) != CSTR_EQUAL)) { 196 err = out_ptr; 197 }else { 198 out_ptr += drive_len; 199 continue; 200 } 201 } else if(exp_ptr+sizeof(path_cmd) <= exp_end 202 && !memcmp(exp_ptr, path_cmd, sizeof(path_cmd))) { 203 exp_ptr += sizeof(path_cmd); 204 if(out_end-out_ptr < path_len 205 || (CompareStringA(LOCALE_SYSTEM_DEFAULT, NORM_IGNORECASE, 206 out_ptr, path_len, path, path_len) != CSTR_EQUAL)) { 207 err = out_ptr; 208 }else { 209 out_ptr += path_len; 210 continue; 211 } 212 } else if(exp_ptr+sizeof(shortpath_cmd) <= exp_end 213 && !memcmp(exp_ptr, shortpath_cmd, sizeof(shortpath_cmd))) { 214 exp_ptr += sizeof(shortpath_cmd); 215 if(out_end-out_ptr < shortpath_len 216 || (CompareStringA(LOCALE_SYSTEM_DEFAULT, NORM_IGNORECASE, 217 out_ptr, shortpath_len, shortpath, shortpath_len) != CSTR_EQUAL)) { 218 err = out_ptr; 219 }else { 220 out_ptr += shortpath_len; 221 continue; 222 } 223 }else if(exp_ptr+sizeof(space_cmd) <= exp_end 224 && !memcmp(exp_ptr, space_cmd, sizeof(space_cmd))) { 225 exp_ptr += sizeof(space_cmd); 226 if(out_ptr < out_end && *out_ptr == ' ') { 227 out_ptr++; 228 continue; 229 } else { 230 err = out_end; 231 } 232 }else if(exp_ptr+sizeof(spaces_cmd) <= exp_end 233 && !memcmp(exp_ptr, spaces_cmd, sizeof(spaces_cmd))) { 234 exp_ptr += sizeof(spaces_cmd); 235 if(out_ptr < out_end && *out_ptr == ' ') { 236 while (out_ptr < out_end && *out_ptr == ' ') out_ptr++; 237 continue; 238 } else { 239 err = out_end; 240 } 241 }else if(exp_ptr+sizeof(tab_cmd) <= exp_end 242 && !memcmp(exp_ptr, tab_cmd, sizeof(tab_cmd))) { 243 exp_ptr += sizeof(tab_cmd); 244 if(out_ptr < out_end && *out_ptr == '\t') { 245 out_ptr++; 246 continue; 247 } else { 248 err = out_end; 249 } 250 }else if(exp_ptr+sizeof(or_broken_cmd) <= exp_end 251 && !memcmp(exp_ptr, or_broken_cmd, sizeof(or_broken_cmd))) { 252 if(out_ptr == out_end) 253 return NULL; 254 else 255 err = out_ptr; 256 }else if(out_ptr == out_end || *out_ptr != *exp_ptr) 257 err = out_ptr; 258 }else if(out_ptr == out_end || *out_ptr != *exp_ptr) { 259 err = out_ptr; 260 } 261 262 if(err) { 263 if(!broken(1)) 264 return err; 265 266 while(exp_ptr+sizeof(or_broken_cmd) <= exp_end && memcmp(exp_ptr, or_broken_cmd, sizeof(or_broken_cmd))) 267 exp_ptr++; 268 exp_ptr += sizeof(or_broken_cmd); 269 if (exp_ptr > exp_end) return err; 270 out_ptr = out_line; 271 err = NULL; 272 continue; 273 } 274 275 exp_ptr++; 276 out_ptr++; 277 } 278 279 if(exp_ptr != exp_end) 280 return out_ptr; 281 else if(out_ptr != out_end) 282 return exp_end; 283 284 return NULL; 285} 286 287static void test_output(const char *out_data, DWORD out_size, const char *exp_data, DWORD exp_size) 288{ 289 const char *out_ptr = out_data, *exp_ptr = exp_data, *out_nl, *exp_nl, *err = NULL; 290 DWORD line = 0; 291 static const char todo_wine_cmd[] = {'@','t','o','d','o','_','w','i','n','e','@'}; 292 static const char resync_cmd[] = {'-','-','-'}; 293 BOOL is_todo_wine, is_out_resync = FALSE, is_exp_resync = FALSE; 294 295 while(out_ptr < out_data+out_size && exp_ptr < exp_data+exp_size) { 296 line++; 297 298 for(exp_nl = exp_ptr; exp_nl < exp_data+exp_size && *exp_nl != '\r' && *exp_nl != '\n'; exp_nl++); 299 for(out_nl = out_ptr; out_nl < out_data+out_size && *out_nl != '\r' && *out_nl != '\n'; out_nl++); 300 301 is_todo_wine = (exp_ptr+sizeof(todo_wine_cmd) <= exp_nl && 302 !memcmp(exp_ptr, todo_wine_cmd, sizeof(todo_wine_cmd))); 303 if (is_todo_wine) 304 exp_ptr += sizeof(todo_wine_cmd); 305 306 todo_wine_if(is_todo_wine) 307 { 308 is_exp_resync=(exp_ptr+sizeof(resync_cmd) <= exp_nl && 309 !memcmp(exp_ptr, resync_cmd, sizeof(resync_cmd))); 310 is_out_resync=(out_ptr+sizeof(resync_cmd) <= out_nl && 311 !memcmp(out_ptr, resync_cmd, sizeof(resync_cmd))); 312 313 err = compare_line(out_ptr, out_nl, exp_ptr, exp_nl); 314 if(err == out_nl) 315 ok(0, "unexpected end of line %d (got '%.*s', wanted '%.*s')\n", 316 line, (int)(out_nl-out_ptr), out_ptr, (int)(exp_nl-exp_ptr), exp_ptr); 317 else if(err == exp_nl) 318 ok(0, "excess characters on line %d (got '%.*s', wanted '%.*s')\n", 319 line, (int)(out_nl-out_ptr), out_ptr, (int)(exp_nl-exp_ptr), exp_ptr); 320 else if (!err && is_todo_wine && is_out_resync && is_exp_resync) 321 /* Consider that the todo_wine was to deal with extra lines, 322 * not for the resync line itself 323 */ 324 err = NULL; 325 else 326 ok(!err, "unexpected char 0x%x position %d in line %d (got '%.*s', wanted '%.*s')\n", 327 (err ? *err : 0), (err ? (int)(err-out_ptr) : -1), line, (int)(out_nl-out_ptr), out_ptr, (int)(exp_nl-exp_ptr), exp_ptr); 328 } 329 330 if (is_exp_resync && err && is_todo_wine) 331 { 332 exp_ptr -= sizeof(todo_wine_cmd); 333 /* If we rewind to the beginning of the line, don't increment line number */ 334 line--; 335 } 336 else if (!is_exp_resync || !err || 337 (is_exp_resync && is_out_resync && err)) 338 { 339 exp_ptr = exp_nl+1; 340 if(exp_nl+1 < exp_data+exp_size && exp_nl[0] == '\r' && exp_nl[1] == '\n') 341 exp_ptr++; 342 } 343 344 if (!is_out_resync || !err) 345 { 346 out_ptr = out_nl+1; 347 if(out_nl+1 < out_data+out_size && out_nl[0] == '\r' && out_nl[1] == '\n') 348 out_ptr++; 349 } 350 } 351 352 ok(exp_ptr >= exp_data+exp_size, "unexpected end of output in line %d, missing %s\n", line, exp_ptr); 353 ok(out_ptr >= out_data+out_size, "too long output, got additional %s\n", out_ptr); 354} 355 356static void run_test(const char *cmd_data, DWORD cmd_size, const char *exp_data, DWORD exp_size) 357{ 358 const char *out_data, *actual_cmd_data; 359 DWORD out_size, actual_cmd_size; 360 361 actual_cmd_data = convert_input_data(cmd_data, cmd_size, &actual_cmd_size); 362 if(!actual_cmd_size || !actual_cmd_data) 363 goto cleanup; 364 365 if(!run_cmd(actual_cmd_data, actual_cmd_size)) 366 goto cleanup; 367 368 out_size = map_file("test.out", &out_data); 369 if(out_size) { 370 test_output(out_data, out_size, exp_data, exp_size); 371 UnmapViewOfFile(out_data); 372 } 373 DeleteFileA("test.out"); 374 DeleteFileA("test.err"); 375 376cleanup: 377 HeapFree(GetProcessHeap(), 0, (LPVOID)actual_cmd_data); 378} 379 380static void run_from_file(const char *file_name) 381{ 382 char out_name[MAX_PATH]; 383 const char *test_data, *out_data; 384 DWORD test_size, out_size; 385 386 test_size = map_file(file_name, &test_data); 387 if(!test_size) { 388 ok(0, "Could not map file %s: %u\n", file_name, GetLastError()); 389 return; 390 } 391 392 sprintf(out_name, "%s.exp", file_name); 393 out_size = map_file(out_name, &out_data); 394 if(!out_size) { 395 ok(0, "Could not map file %s: %u\n", out_name, GetLastError()); 396 UnmapViewOfFile(test_data); 397 return; 398 } 399 400 run_test(test_data, test_size, out_data, out_size); 401 402 UnmapViewOfFile(test_data); 403 UnmapViewOfFile(out_data); 404} 405 406static DWORD load_resource(const char *name, const char *type, const char **ret) 407{ 408 const char *res; 409 HRSRC src; 410 DWORD size; 411 412 src = FindResourceA(NULL, name, type); 413 ok(src != NULL, "Could not find resource %s: %u\n", name, GetLastError()); 414 if(!src) 415 return 0; 416 417 res = LoadResource(NULL, src); 418 size = SizeofResource(NULL, src); 419 while(size && !res[size-1]) 420 size--; 421 422 *ret = res; 423 return size; 424} 425 426static BOOL WINAPI test_enum_proc(HMODULE module, LPCSTR type, LPSTR name, LONG_PTR param) 427{ 428 const char *cmd_data, *out_data; 429 DWORD cmd_size, out_size; 430 char res_name[100]; 431 432 trace("running %s test...\n", name); 433 434 cmd_size = load_resource(name, type, &cmd_data); 435 if(!cmd_size) 436 return TRUE; 437 438 sprintf(res_name, "%s.exp", name); 439 out_size = load_resource(res_name, "TESTOUT", &out_data); 440 if(!out_size) 441 return TRUE; 442 443 run_test(cmd_data, cmd_size, out_data, out_size); 444 return TRUE; 445} 446 447static int cmd_available(void) 448{ 449 STARTUPINFOA si; 450 PROCESS_INFORMATION pi; 451 char cmd[] = "cmd /c exit 0"; 452 453 memset(&si, 0, sizeof(si)); 454 si.cb = sizeof(si); 455 if (CreateProcessA(NULL, cmd, NULL, NULL, FALSE, 0, NULL, NULL, &si, &pi)) { 456 CloseHandle(pi.hThread); 457 CloseHandle(pi.hProcess); 458 return TRUE; 459 } 460 return FALSE; 461} 462 463START_TEST(reactos) 464{ 465 int argc; 466 char **argv; 467 468 if (!cmd_available()) { 469 win_skip("cmd not installed, skipping cmd tests\n"); 470 return; 471 } 472 473 workdir_len = GetCurrentDirectoryA(sizeof(workdir), workdir); 474 drive[0] = workdir[0]; 475 drive[1] = workdir[1]; /* Should be ':' */ 476 memcpy(path, workdir + drive_len, (workdir_len - drive_len) * sizeof(drive[0])); 477 478 /* Only add trailing backslash to 'path' for non-root directory */ 479 if (workdir_len - drive_len > 1) { 480 path[workdir_len - drive_len] = '\\'; 481 path_len = workdir_len - drive_len + 1; 482 } else { 483 path_len = 1; /* \ */ 484 } 485 shortpath_len = GetShortPathNameA(path, shortpath, ARRAY_SIZE(shortpath)); 486 487 argc = winetest_get_mainargs(&argv); 488 if(argc > 2) 489 run_from_file(argv[2]); 490 else 491 EnumResourceNamesA(NULL, "TESTCMD", test_enum_proc, 0); 492}