2 * Unit test suite for localspl API functions: local print monitor
4 * Copyright 2006-2007 Detlef Riekenberg
6 * 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.
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.
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
18 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
31 #include "ddk/winsplp.h"
33 #include "wine/test.h"
39 static HMODULE hlocalmon
;
40 static LPMONITOREX (WINAPI
*pInitializePrintMonitor
)(LPWSTR
);
42 static LPMONITOREX pm
;
43 static BOOL (WINAPI
*pEnumPorts
)(LPWSTR
, DWORD
, LPBYTE
, DWORD
, LPDWORD
, LPDWORD
);
44 static BOOL (WINAPI
*pOpenPort
)(LPWSTR
, PHANDLE
);
45 static BOOL (WINAPI
*pOpenPortEx
)(LPWSTR
, LPWSTR
, PHANDLE
, struct _MONITOR
*);
46 static BOOL (WINAPI
*pStartDocPort
)(HANDLE
, LPWSTR
, DWORD
, DWORD
, LPBYTE
);
47 static BOOL (WINAPI
*pWritePort
)(HANDLE hPort
, LPBYTE
, DWORD
, LPDWORD
);
48 static BOOL (WINAPI
*pReadPort
)(HANDLE hPort
, LPBYTE
, DWORD
, LPDWORD
);
49 static BOOL (WINAPI
*pEndDocPort
)(HANDLE
);
50 static BOOL (WINAPI
*pClosePort
)(HANDLE
);
51 static BOOL (WINAPI
*pAddPort
)(LPWSTR
, HWND
, LPWSTR
);
52 static BOOL (WINAPI
*pAddPortEx
)(LPWSTR
, DWORD
, LPBYTE
, LPWSTR
);
53 static BOOL (WINAPI
*pConfigurePort
)(LPWSTR
, HWND
, LPWSTR
);
54 static BOOL (WINAPI
*pDeletePort
)(LPWSTR
, HWND
, LPWSTR
);
55 static BOOL (WINAPI
*pGetPrinterDataFromPort
)(HANDLE
, DWORD
, LPWSTR
, LPWSTR
, DWORD
, LPWSTR
, DWORD
, LPDWORD
);
56 static BOOL (WINAPI
*pSetPortTimeOuts
)(HANDLE
, LPCOMMTIMEOUTS
, DWORD
);
57 static BOOL (WINAPI
*pXcvOpenPort
)(LPCWSTR
, ACCESS_MASK
, PHANDLE phXcv
);
58 static DWORD (WINAPI
*pXcvDataPort
)(HANDLE
, LPCWSTR
, PBYTE
, DWORD
, PBYTE
, DWORD
, PDWORD
);
59 static BOOL (WINAPI
*pXcvClosePort
)(HANDLE
);
62 static HANDLE hXcv_noaccess
;
64 /* ########################### */
66 static const WCHAR cmd_AddPortW
[] = {'A','d','d','P','o','r','t',0};
67 static const WCHAR cmd_ConfigureLPTPortCommandOKW
[] = {'C','o','n','f','i','g','u','r','e',
68 'L','P','T','P','o','r','t',
69 'C','o','m','m','a','n','d','O','K',0};
70 static WCHAR cmd_DeletePortW
[] = {'D','e','l','e','t','e','P','o','r','t',0};
71 static WCHAR cmd_GetTransmissionRetryTimeoutW
[] = {'G','e','t',
72 'T','r','a','n','s','m','i','s','s','i','o','n',
73 'R','e','t','r','y','T','i','m','e','o','u','t',0};
75 static WCHAR cmd_MonitorUIW
[] = {'M','o','n','i','t','o','r','U','I',0};
76 static WCHAR cmd_MonitorUI_lcaseW
[] = {'m','o','n','i','t','o','r','u','i',0};
77 static WCHAR cmd_PortIsValidW
[] = {'P','o','r','t','I','s','V','a','l','i','d',0};
78 static WCHAR does_not_existW
[] = {'d','o','e','s','_','n','o','t','_','e','x','i','s','t',0};
79 static CHAR emptyA
[] = "";
80 static WCHAR emptyW
[] = {0};
81 static WCHAR LocalPortW
[] = {'L','o','c','a','l',' ','P','o','r','t',0};
82 static WCHAR Monitors_LocalPortW
[] = {
83 'S','y','s','t','e','m','\\',
84 'C','u','r','r','e','n','t','C','o','n','t','r','o','l','S','e','t','\\',
85 'C','o','n','t','r','o','l','\\',
86 'P','r','i','n','t','\\',
87 'M','o','n','i','t','o','r','s','\\',
88 'L','o','c','a','l',' ','P','o','r','t',0};
90 static CHAR num_0A
[] = "0";
91 static WCHAR num_0W
[] = {'0',0};
92 static CHAR num_1A
[] = "1";
93 static WCHAR num_1W
[] = {'1',0};
94 static CHAR num_999999A
[] = "999999";
95 static WCHAR num_999999W
[] = {'9','9','9','9','9','9',0};
96 static CHAR num_1000000A
[] = "1000000";
97 static WCHAR num_1000000W
[] = {'1','0','0','0','0','0','0',0};
99 static WCHAR portname_comW
[] = {'C','O','M',0};
100 static WCHAR portname_com1W
[] = {'C','O','M','1',':',0};
101 static WCHAR portname_com2W
[] = {'C','O','M','2',':',0};
102 static WCHAR portname_fileW
[] = {'F','I','L','E',':',0};
103 static WCHAR portname_lptW
[] = {'L','P','T',0};
104 static WCHAR portname_lpt1W
[] = {'L','P','T','1',':',0};
105 static WCHAR portname_lpt2W
[] = {'L','P','T','2',':',0};
106 static WCHAR server_does_not_existW
[] = {'\\','\\','d','o','e','s','_','n','o','t','_','e','x','i','s','t',0};
108 static CHAR TransmissionRetryTimeoutA
[] = {'T','r','a','n','s','m','i','s','s','i','o','n',
109 'R','e','t','r','y','T','i','m','e','o','u','t',0};
111 static CHAR WinNT_CV_WindowsA
[] = {'S','o','f','t','w','a','r','e','\\',
112 'M','i','c','r','o','s','o','f','t','\\',
113 'W','i','n','d','o','w','s',' ','N','T','\\',
114 'C','u','r','r','e','n','t','V','e','r','s','i','o','n','\\',
115 'W','i','n','d','o','w','s',0};
116 static WCHAR wineW
[] = {'W','i','n','e',0};
118 static WCHAR tempdirW
[MAX_PATH
];
119 static WCHAR tempfileW
[MAX_PATH
];
121 #define PORTNAME_PREFIX 3
122 #define PORTNAME_MINSIZE 5
123 #define PORTNAME_MAXSIZE 10
124 static WCHAR have_com
[PORTNAME_MAXSIZE
];
125 static WCHAR have_lpt
[PORTNAME_MAXSIZE
];
126 static WCHAR have_file
[PORTNAME_MAXSIZE
];
128 /* ########################### */
130 static DWORD
delete_port(LPWSTR portname
)
135 res
= pDeletePort(NULL
, 0, portname
);
139 res
= pXcvDataPort(hXcv
, cmd_DeletePortW
, (PBYTE
) portname
, (lstrlenW(portname
) + 1) * sizeof(WCHAR
), NULL
, 0, NULL
);
144 /* ########################### */
146 static void find_installed_ports(void)
148 PORT_INFO_1W
* pi
= NULL
;
149 WCHAR nameW
[PORTNAME_MAXSIZE
];
159 if (!pEnumPorts
) return;
161 res
= pEnumPorts(NULL
, 1, NULL
, 0, &needed
, &returned
);
162 if (!res
&& (GetLastError() == ERROR_INSUFFICIENT_BUFFER
)) {
163 pi
= HeapAlloc(GetProcessHeap(), 0, needed
);
165 res
= pEnumPorts(NULL
, 1, (LPBYTE
) pi
, needed
, &needed
, &returned
);
168 skip("no ports found\n");
169 HeapFree(GetProcessHeap(), 0, pi
);
174 while (id
< returned
) {
175 res
= lstrlenW(pi
[id
].pName
);
176 if ((res
>= PORTNAME_MINSIZE
) && (res
< PORTNAME_MAXSIZE
) &&
177 (pi
[id
].pName
[res
-1] == ':')) {
178 /* copy only the prefix ("LPT" or "COM") */
179 memcpy(&nameW
, pi
[id
].pName
, PORTNAME_PREFIX
* sizeof(WCHAR
));
180 nameW
[PORTNAME_PREFIX
] = '\0';
182 if (!have_com
[0] && (lstrcmpiW(nameW
, portname_comW
) == 0)) {
183 memcpy(&have_com
, pi
[id
].pName
, (res
+1) * sizeof(WCHAR
));
186 if (!have_lpt
[0] && (lstrcmpiW(nameW
, portname_lptW
) == 0)) {
187 memcpy(&have_lpt
, pi
[id
].pName
, (res
+1) * sizeof(WCHAR
));
190 if (!have_file
[0] && (lstrcmpiW(pi
[id
].pName
, portname_fileW
) == 0)) {
191 memcpy(&have_file
, pi
[id
].pName
, (res
+1) * sizeof(WCHAR
));
197 HeapFree(GetProcessHeap(), 0, pi
);
200 /* ########################### */
202 static void test_AddPort(void)
206 /* moved to localui.dll since w2k */
207 if (!pAddPort
) return;
211 /* NT4 crash on this test */
212 res
= pAddPort(NULL
, 0, NULL
);
215 /* Testing-Results (localmon.dll from NT4.0):
216 - The Servername is ignored
217 - Case of MonitorName is ignored
220 SetLastError(0xdeadbeef);
221 res
= pAddPort(NULL
, 0, emptyW
);
222 ok(!res
, "returned %d with %u (expected '0')\n", res
, GetLastError());
224 SetLastError(0xdeadbeef);
225 res
= pAddPort(NULL
, 0, does_not_existW
);
226 ok(!res
, "returned %d with %u (expected '0')\n", res
, GetLastError());
230 /* ########################### */
232 static void test_AddPortEx(void)
241 if ((!pDeletePort
) && (!hXcv
)) {
242 skip("No API to delete a Port\n");
246 /* start test with clean ports */
247 delete_port(tempfileW
);
249 pi
.pPortName
= tempfileW
;
251 /* tests crash with native localspl.dll in w2k,
252 but works with native localspl.dll in wine */
253 SetLastError(0xdeadbeef);
254 res
= pAddPortEx(NULL
, 1, (LPBYTE
) &pi
, LocalPortW
);
255 trace("returned %u with %u\n", res
, GetLastError() );
256 ok( res
, "got %u with %u (expected '!= 0')\n", res
, GetLastError());
258 /* port already exists: */
259 SetLastError(0xdeadbeef);
260 res
= pAddPortEx(NULL
, 1, (LPBYTE
) &pi
, LocalPortW
);
261 trace("returned %u with %u\n", res
, GetLastError() );
262 ok( !res
&& (GetLastError() == ERROR_INVALID_PARAMETER
),
263 "got %u with %u (expected '0' with ERROR_INVALID_PARAMETER)\n",
264 res
, GetLastError());
265 delete_port(tempfileW
);
268 /* NULL for pMonitorName is documented for Printmonitors, but
269 localspl.dll fails always with ERROR_INVALID_PARAMETER */
270 SetLastError(0xdeadbeef);
271 res
= pAddPortEx(NULL
, 1, (LPBYTE
) &pi
, NULL
);
272 trace("returned %u with %u\n", res
, GetLastError() );
273 ok( !res
&& (GetLastError() == ERROR_INVALID_PARAMETER
),
274 "got %u with %u (expected '0' with ERROR_INVALID_PARAMETER)\n",
275 res
, GetLastError());
276 if (res
) delete_port(tempfileW
);
279 SetLastError(0xdeadbeef);
280 res
= pAddPortEx(NULL
, 1, (LPBYTE
) &pi
, emptyW
);
281 trace("returned %u with %u\n", res
, GetLastError() );
282 ok( !res
&& (GetLastError() == ERROR_INVALID_PARAMETER
),
283 "got %u with %u (expected '0' with ERROR_INVALID_PARAMETER)\n",
284 res
, GetLastError());
285 if (res
) delete_port(tempfileW
);
288 SetLastError(0xdeadbeef);
289 res
= pAddPortEx(NULL
, 1, (LPBYTE
) &pi
, does_not_existW
);
290 trace("returned %u with %u\n", res
, GetLastError() );
291 ok( !res
&& (GetLastError() == ERROR_INVALID_PARAMETER
),
292 "got %u with %u (expected '0' with ERROR_INVALID_PARAMETER)\n",
293 res
, GetLastError());
294 if (res
) delete_port(tempfileW
);
298 SetLastError(0xdeadbeef);
299 res
= pAddPortEx(NULL
, 1, (LPBYTE
) &pi
, LocalPortW
);
300 ok( !res
&& (GetLastError() == ERROR_INVALID_PARAMETER
),
301 "got %u with %u (expected '0' with ERROR_INVALID_PARAMETER)\n",
302 res
, GetLastError());
304 /* level 2 is documented as supported for Printmonitors,
305 but localspl.dll fails always with ERROR_INVALID_LEVEL */
307 pi
.pPortName
= tempfileW
;
308 pi
.pMonitorName
= LocalPortW
;
309 pi
.pDescription
= wineW
;
310 pi
.fPortType
= PORT_TYPE_WRITE
;
312 SetLastError(0xdeadbeef);
313 res
= pAddPortEx(NULL
, 2, (LPBYTE
) &pi
, LocalPortW
);
314 ok( !res
&& (GetLastError() == ERROR_INVALID_LEVEL
),
315 "got %u with %u (expected '0' with ERROR_INVALID_LEVEL)\n",
316 res
, GetLastError());
317 if (res
) delete_port(tempfileW
);
321 SetLastError(0xdeadbeef);
322 res
= pAddPortEx(NULL
, 0, (LPBYTE
) &pi
, LocalPortW
);
323 ok( !res
&& (GetLastError() == ERROR_INVALID_LEVEL
),
324 "got %u with %u (expected '0' with ERROR_INVALID_LEVEL)\n",
325 res
, GetLastError());
326 if (res
) delete_port(tempfileW
);
329 SetLastError(0xdeadbeef);
330 res
= pAddPortEx(NULL
, 3, (LPBYTE
) &pi
, LocalPortW
);
331 ok( !res
&& (GetLastError() == ERROR_INVALID_LEVEL
),
332 "got %u with %u (expected '0' with ERROR_INVALID_LEVEL)\n",
333 res
, GetLastError());
334 if (res
) delete_port(tempfileW
);
337 delete_port(tempfileW
);
340 /* ########################### */
342 static void test_ClosePort(void)
351 if (!pOpenPort
|| !pClosePort
) return;
356 hPort
= (HANDLE
) 0xdeadbeef;
357 res
= pOpenPort(nameW
, &hPort
);
358 hPort2
= (HANDLE
) 0xdeadbeef;
359 res2
= pOpenPort(nameW
, &hPort2
);
361 if (res2
&& (hPort2
!= hPort
)) {
362 SetLastError(0xdeadbeef);
363 res2
= pClosePort(hPort2
);
364 ok(res2
, "got %u with %u (expected '!= 0')\n", res2
, GetLastError());
368 SetLastError(0xdeadbeef);
369 res
= pClosePort(hPort
);
370 ok(res
, "got %u with %u (expected '!= 0')\n", res
, GetLastError());
378 hPort
= (HANDLE
) 0xdeadbeef;
379 res
= pOpenPort(nameW
, &hPort
);
380 hPort2
= (HANDLE
) 0xdeadbeef;
381 res2
= pOpenPort(nameW
, &hPort2
);
383 if (res2
&& (hPort2
!= hPort
)) {
384 SetLastError(0xdeadbeef);
385 res2
= pClosePort(hPort2
);
386 ok(res2
, "got %u with %u (expected '!= 0')\n", res2
, GetLastError());
390 SetLastError(0xdeadbeef);
391 res
= pClosePort(hPort
);
392 ok(res
, "got %u with %u (expected '!= 0')\n", res
, GetLastError());
400 hPort
= (HANDLE
) 0xdeadbeef;
401 res
= pOpenPort(nameW
, &hPort
);
402 hPort2
= (HANDLE
) 0xdeadbeef;
403 res2
= pOpenPort(nameW
, &hPort2
);
405 if (res2
&& (hPort2
!= hPort
)) {
406 SetLastError(0xdeadbeef);
407 res2
= pClosePort(hPort2
);
408 ok(res2
, "got %u with %u (expected '!= 0')\n", res2
, GetLastError());
412 SetLastError(0xdeadbeef);
413 res
= pClosePort(hPort
);
414 ok(res
, "got %u with %u (expected '!= 0')\n", res
, GetLastError());
420 /* an invalid HANDLE crash native localspl.dll */
422 SetLastError(0xdeadbeef);
423 res
= pClosePort(NULL
);
424 trace("got %u with %u\n", res
, GetLastError());
426 SetLastError(0xdeadbeef);
427 res
= pClosePort( (HANDLE
) 0xdeadbeef);
428 trace("got %u with %u\n", res
, GetLastError());
430 SetLastError(0xdeadbeef);
431 res
= pClosePort(INVALID_HANDLE_VALUE
);
432 trace("got %u with %u\n", res
, GetLastError());
437 /* ########################### */
439 static void test_ConfigurePort(void)
443 /* moved to localui.dll since w2k */
444 if (!pConfigurePort
) return;
448 /* NT4 crash on this test */
449 res
= pConfigurePort(NULL
, 0, NULL
);
452 /* Testing-Results (localmon.dll from NT4.0):
453 - Case of Portname is ignored
454 - "COM1:" and "COM01:" are the same (Compared by value)
455 - Portname without ":" => Dialog "Nothing to configure" comes up; Success
456 - "LPT1:", "LPT0:" and "LPT:" are the same (Numbers in "LPT:" are ignored)
457 - Empty Servername (LPT1:) => Dialog comes up (Servername is ignored)
458 - "FILE:" => Dialog "Nothing to configure" comes up; Success
459 - Empty Portname => => Dialog "Nothing to configure" comes up; Success
460 - Port "does_not_exist" => Dialog "Nothing to configure" comes up; Success
462 if (winetest_interactive
> 0) {
464 SetLastError(0xdeadbeef);
465 res
= pConfigurePort(NULL
, 0, portname_com1W
);
466 trace("returned %d with %u\n", res
, GetLastError());
468 SetLastError(0xdeadbeef);
469 res
= pConfigurePort(NULL
, 0, portname_lpt1W
);
470 trace("returned %d with %u\n", res
, GetLastError());
472 SetLastError(0xdeadbeef);
473 res
= pConfigurePort(NULL
, 0, portname_fileW
);
474 trace("returned %d with %u\n", res
, GetLastError());
478 /* ########################### */
480 static void test_DeletePort(void)
484 /* moved to localui.dll since w2k */
485 if (!pDeletePort
) return;
489 /* NT4 crash on this test */
490 res
= pDeletePort(NULL
, 0, NULL
);
493 /* Testing-Results (localmon.dll from NT4.0):
494 - Case of Portname is ignored (returned '1' on Success)
495 - "COM1:" and "COM01:" are different (Compared as string)
496 - server_does_not_exist (LPT1:) => Port deleted, Success (Servername is ignored)
497 - Empty Portname => => FALSE (LastError not changed)
498 - Port "does_not_exist" => FALSE (LastError not changed)
501 SetLastError(0xdeadbeef);
502 res
= pDeletePort(NULL
, 0, emptyW
);
503 ok(!res
, "returned %d with %u (expected '0')\n", res
, GetLastError());
505 SetLastError(0xdeadbeef);
506 res
= pDeletePort(NULL
, 0, does_not_existW
);
507 ok(!res
, "returned %d with %u (expected '0')\n", res
, GetLastError());
511 /* ########################### */
513 static void test_EnumPorts(void)
522 if (!pEnumPorts
) return;
524 /* valid levels are 1 and 2 */
525 for(level
= 0; level
< 4; level
++) {
528 pcReturned
= 0xdeadbeef;
529 SetLastError(0xdeadbeef);
530 res
= pEnumPorts(NULL
, level
, NULL
, 0, &cbBuf
, &pcReturned
);
532 /* use only a short test, when we test with an invalid level */
533 if(!level
|| (level
> 2)) {
534 /* NT4 fails with ERROR_INVALID_LEVEL (as expected)
535 XP succeeds with ERROR_SUCCESS () */
536 ok( (cbBuf
== 0) && (pcReturned
== 0),
537 "(%d) returned %d with %u and %d, %d (expected 0, 0)\n",
538 level
, res
, GetLastError(), cbBuf
, pcReturned
);
542 ok( !res
&& (GetLastError() == ERROR_INSUFFICIENT_BUFFER
),
543 "(%d) returned %d with %u and %d, %d (expected '0' with "
544 "ERROR_INSUFFICIENT_BUFFER)\n",
545 level
, res
, GetLastError(), cbBuf
, pcReturned
);
547 buffer
= HeapAlloc(GetProcessHeap(), 0, cbBuf
* 2);
548 if (buffer
== NULL
) continue;
550 pcbNeeded
= 0xdeadbeef;
551 pcReturned
= 0xdeadbeef;
552 SetLastError(0xdeadbeef);
553 res
= pEnumPorts(NULL
, level
, buffer
, cbBuf
, &pcbNeeded
, &pcReturned
);
554 ok( res
, "(%d) returned %d with %u and %d, %d (expected '!= 0')\n",
555 level
, res
, GetLastError(), pcbNeeded
, pcReturned
);
556 /* We can compare the returned Data with the Registry / "win.ini",[Ports] here */
558 pcbNeeded
= 0xdeadbeef;
559 pcReturned
= 0xdeadbeef;
560 SetLastError(0xdeadbeef);
561 res
= pEnumPorts(NULL
, level
, buffer
, cbBuf
+1, &pcbNeeded
, &pcReturned
);
562 ok( res
, "(%d) returned %d with %u and %d, %d (expected '!= 0')\n",
563 level
, res
, GetLastError(), pcbNeeded
, pcReturned
);
565 pcbNeeded
= 0xdeadbeef;
566 pcReturned
= 0xdeadbeef;
567 SetLastError(0xdeadbeef);
568 res
= pEnumPorts(NULL
, level
, buffer
, cbBuf
-1, &pcbNeeded
, &pcReturned
);
569 ok( !res
&& (GetLastError() == ERROR_INSUFFICIENT_BUFFER
),
570 "(%d) returned %d with %u and %d, %d (expected '0' with "
571 "ERROR_INSUFFICIENT_BUFFER)\n",
572 level
, res
, GetLastError(), pcbNeeded
, pcReturned
);
576 /* The following tests crash this app with native localmon/localspl */
577 res
= pEnumPorts(NULL
, level
, NULL
, cbBuf
, &pcbNeeded
, &pcReturned
);
578 res
= pEnumPorts(NULL
, level
, buffer
, cbBuf
, NULL
, &pcReturned
);
579 res
= pEnumPorts(NULL
, level
, buffer
, cbBuf
, &pcbNeeded
, NULL
);
582 /* The Servername is ignored */
583 pcbNeeded
= 0xdeadbeef;
584 pcReturned
= 0xdeadbeef;
585 SetLastError(0xdeadbeef);
586 res
= pEnumPorts(emptyW
, level
, buffer
, cbBuf
+1, &pcbNeeded
, &pcReturned
);
587 ok( res
, "(%d) returned %d with %u and %d, %d (expected '!= 0')\n",
588 level
, res
, GetLastError(), pcbNeeded
, pcReturned
);
590 pcbNeeded
= 0xdeadbeef;
591 pcReturned
= 0xdeadbeef;
592 SetLastError(0xdeadbeef);
593 res
= pEnumPorts(server_does_not_existW
, level
, buffer
, cbBuf
+1, &pcbNeeded
, &pcReturned
);
594 ok( res
, "(%d) returned %d with %u and %d, %d (expected '!= 0')\n",
595 level
, res
, GetLastError(), pcbNeeded
, pcReturned
);
597 HeapFree(GetProcessHeap(), 0, buffer
);
601 /* ########################### */
604 static void test_InitializePrintMonitor(void)
608 SetLastError(0xdeadbeef);
609 res
= pInitializePrintMonitor(NULL
);
610 /* The Parameter was unchecked before w2k */
611 ok( res
|| (GetLastError() == ERROR_INVALID_PARAMETER
),
612 "returned %p with %u\n (expected '!= NULL' or: NULL with "
613 "ERROR_INVALID_PARAMETER)\n", res
, GetLastError());
615 SetLastError(0xdeadbeef);
616 res
= pInitializePrintMonitor(emptyW
);
617 ok( res
|| (GetLastError() == ERROR_INVALID_PARAMETER
),
618 "returned %p with %u\n (expected '!= NULL' or: NULL with "
619 "ERROR_INVALID_PARAMETER)\n", res
, GetLastError());
622 /* Every call with a non-empty string returns the same Pointer */
623 SetLastError(0xdeadbeef);
624 res
= pInitializePrintMonitor(Monitors_LocalPortW
);
626 "returned %p with %u (expected %p)\n", res
, GetLastError(), pm
);
630 /* ########################### */
632 static void test_OpenPort(void)
640 if (!pOpenPort
|| !pClosePort
) return;
645 hPort
= (HANDLE
) 0xdeadbeef;
646 SetLastError(0xdeadbeef);
647 res
= pOpenPort(nameW
, &hPort
);
648 ok( res
, "got %u with %u and %p (expected '!= 0')\n",
649 res
, GetLastError(), hPort
);
651 /* the same HANDLE is returned for a second OpenPort in native localspl */
652 hPort2
= (HANDLE
) 0xdeadbeef;
653 SetLastError(0xdeadbeef);
654 res2
= pOpenPort(nameW
, &hPort2
);
655 ok( res2
, "got %u with %u and %p (expected '!= 0')\n",
656 res2
, GetLastError(), hPort2
);
658 if (res
) pClosePort(hPort
);
659 if (res2
&& (hPort2
!= hPort
)) pClosePort(hPort2
);
665 hPort
= (HANDLE
) 0xdeadbeef;
666 SetLastError(0xdeadbeef);
667 res
= pOpenPort(nameW
, &hPort
);
668 ok( res
|| (GetLastError() == ERROR_ACCESS_DENIED
),
669 "got %u with %u and %p (expected '!= 0' or '0' with ERROR_ACCESS_DENIED)\n",
670 res
, GetLastError(), hPort
);
672 /* the same HANDLE is returned for a second OpenPort in native localspl */
673 hPort2
= (HANDLE
) 0xdeadbeef;
674 SetLastError(0xdeadbeef);
675 res2
= pOpenPort(nameW
, &hPort2
);
676 ok( res2
|| (GetLastError() == ERROR_ACCESS_DENIED
),
677 "got %u with %u and %p (expected '!= 0' or '0' with ERROR_ACCESS_DENIED)\n",
678 res2
, GetLastError(), hPort2
);
680 if (res
) pClosePort(hPort
);
681 if (res2
&& (hPort2
!= hPort
)) pClosePort(hPort2
);
687 hPort
= (HANDLE
) 0xdeadbeef;
688 SetLastError(0xdeadbeef);
689 res
= pOpenPort(nameW
, &hPort
);
690 ok( res
, "got %u with %u and %p (expected '!= 0')\n",
691 res
, GetLastError(), hPort
);
693 /* a different HANDLE is returned for a second OpenPort */
694 hPort2
= (HANDLE
) 0xdeadbeef;
695 SetLastError(0xdeadbeef);
696 res2
= pOpenPort(nameW
, &hPort2
);
697 ok( res2
&& (hPort2
!= hPort
),
698 "got %u with %u and %p (expected '!= 0' and '!= %p')\n",
699 res2
, GetLastError(), hPort2
, hPort
);
701 if (res
) pClosePort(hPort
);
702 if (res2
&& (hPort2
!= hPort
)) pClosePort(hPort2
);
706 /* this test crash native localspl (w2k+xp) */
708 hPort
= (HANDLE
) 0xdeadbeef;
709 SetLastError(0xdeadbeef);
710 res
= pOpenPort(nameW
, NULL
);
711 trace("got %u with %u and %p\n", res
, GetLastError(), hPort
);
715 hPort
= (HANDLE
) 0xdeadbeef;
716 SetLastError(0xdeadbeef);
717 res
= pOpenPort(does_not_existW
, &hPort
);
718 ok (!res
&& (hPort
== (HANDLE
) 0xdeadbeef),
719 "got %u with 0x%x and %p (expected '0' and 0xdeadbeef)\n", res
, GetLastError(), hPort
);
720 if (res
) pClosePort(hPort
);
722 hPort
= (HANDLE
) 0xdeadbeef;
723 SetLastError(0xdeadbeef);
724 res
= pOpenPort(emptyW
, &hPort
);
725 ok (!res
&& (hPort
== (HANDLE
) 0xdeadbeef),
726 "got %u with 0x%x and %p (expected '0' and 0xdeadbeef)\n", res
, GetLastError(), hPort
);
727 if (res
) pClosePort(hPort
);
730 /* NULL as name crash native localspl (w2k+xp) */
732 hPort
= (HANDLE
) 0xdeadbeef;
733 SetLastError(0xdeadbeef);
734 res
= pOpenPort(NULL
, &hPort
);
735 trace("got %u with %u and %p\n", res
, GetLastError(), hPort
);
740 /* ########################### */
742 static void test_XcvClosePort(void)
750 /* crash with native localspl.dll (w2k+xp) */
751 res
= pXcvClosePort(NULL
);
752 res
= pXcvClosePort(INVALID_HANDLE_VALUE
);
756 SetLastError(0xdeadbeef);
757 hXcv2
= (HANDLE
) 0xdeadbeef;
758 res
= pXcvOpenPort(emptyW
, SERVER_ACCESS_ADMINISTER
, &hXcv2
);
759 ok(res
, "returned %d with %u and %p (expected '!= 0')\n", res
, GetLastError(), hXcv2
);
762 SetLastError(0xdeadbeef);
763 res
= pXcvClosePort(hXcv2
);
764 ok( res
, "returned %d with %u (expected '!= 0')\n", res
, GetLastError());
768 /* test for "Double Free": crash with native localspl.dll (w2k+xp) */
769 res
= pXcvClosePort(hXcv2
);
774 /* ########################### */
776 static void test_XcvDataPort_AddPort(void)
781 * The following tests crash with native localspl.dll on w2k and xp,
782 * but it works, when the native dll (w2k and xp) is used in wine.
783 * also tested (same crash): replacing emptyW with portname_lpt1W
784 * and replacing "NULL, 0, NULL" with "buffer, MAX_PATH, &needed"
786 * We need to use a different API (AddPortEx) instead
790 /* create a Port for a normal, writable file */
791 SetLastError(0xdeadbeef);
792 res
= pXcvDataPort(hXcv
, cmd_AddPortW
, (PBYTE
) tempfileW
, (lstrlenW(tempfileW
) + 1) * sizeof(WCHAR
), NULL
, 0, NULL
);
793 ok( res
== ERROR_SUCCESS
, "returned %d with %u (expected ERROR_SUCCESS)\n", res
, GetLastError());
795 /* add our testport again */
796 SetLastError(0xdeadbeef);
797 res
= pXcvDataPort(hXcv
, cmd_AddPortW
, (PBYTE
) tempfileW
, (lstrlenW(tempfileW
) + 1) * sizeof(WCHAR
), NULL
, 0, NULL
);
798 ok( res
== ERROR_ALREADY_EXISTS
, "returned %d with %u (expected ERROR_ALREADY_EXISTS)\n", res
, GetLastError());
800 /* create a well-known Port */
801 SetLastError(0xdeadbeef);
802 res
= pXcvDataPort(hXcv
, cmd_AddPortW
, (PBYTE
) portname_lpt1W
, (lstrlenW(portname_lpt1W
) + 1) * sizeof(WCHAR
), NULL
, 0, NULL
);
803 ok( res
== ERROR_ALREADY_EXISTS
, "returned %d with %u (expected ERROR_ALREADY_EXISTS)\n", res
, GetLastError());
805 /* ERROR_ALREADY_EXISTS is also returned from native localspl.dll on wine,
806 when "RPT1:" was already installed for redmonnt.dll:
807 res = pXcvDataPort(hXcv, cmd_AddPortW, (PBYTE) portname_rpt1W, ...
811 SetLastError(0xdeadbeef);
812 res
= pXcvDataPort(hXcv
, cmd_DeletePortW
, (PBYTE
) tempfileW
, (lstrlenW(tempfileW
) + 1) * sizeof(WCHAR
), NULL
, 0, NULL
);
813 ok( res
== ERROR_SUCCESS
, "returned %d with %u (expected ERROR_SUCCESS)\n", res
, GetLastError());
818 /* ########################### */
820 static void test_XcvDataPort_ConfigureLPTPortCommandOK(void)
829 /* Read the original value from the registry */
830 res
= RegOpenKeyExA(HKEY_LOCAL_MACHINE
, WinNT_CV_WindowsA
, 0, KEY_ALL_ACCESS
, &hroot
);
831 if (res
== ERROR_ACCESS_DENIED
) {
832 skip("ACCESS_DENIED\n");
836 if (res
!= ERROR_SUCCESS
) {
837 /* unable to open the registry: skip the test */
838 skip("got %d\n", res
);
842 needed
= sizeof(org_value
)-1 ;
843 res
= RegQueryValueExA(hroot
, TransmissionRetryTimeoutA
, NULL
, NULL
, (PBYTE
) org_value
, &needed
);
844 ok( (res
== ERROR_SUCCESS
) || (res
== ERROR_FILE_NOT_FOUND
),
845 "returned %u and %u for \"%s\" (expected ERROR_SUCCESS or "
846 "ERROR_FILE_NOT_FOUND)\n", res
, needed
, org_value
);
848 RegDeleteValueA(hroot
, TransmissionRetryTimeoutA
);
851 needed
= (DWORD
) 0xdeadbeef;
852 SetLastError(0xdeadbeef);
853 res
= pXcvDataPort(hXcv
, cmd_ConfigureLPTPortCommandOKW
, (PBYTE
) num_0W
, sizeof(num_0W
), NULL
, 0, &needed
);
854 if (res
== ERROR_INVALID_PARAMETER
) {
855 skip("'ConfigureLPTPortCommandOK' not supported\n");
858 ok( res
== ERROR_SUCCESS
, "returned %d with %u (expected ERROR_SUCCESS)\n", res
, GetLastError());
859 needed
= sizeof(buffer
)-1 ;
860 res
= RegQueryValueExA(hroot
, TransmissionRetryTimeoutA
, NULL
, NULL
, (PBYTE
) buffer
, &needed
);
861 ok( (res
== ERROR_SUCCESS
) && (lstrcmpA(buffer
, num_0A
) == 0),
862 "returned %d and '%s' (expected ERROR_SUCCESS and '%s')\n",
863 res
, buffer
, num_0A
);
867 needed
= (DWORD
) 0xdeadbeef;
868 SetLastError(0xdeadbeef);
869 res
= pXcvDataPort(hXcv
, cmd_ConfigureLPTPortCommandOKW
, (PBYTE
) num_1W
, sizeof(num_1W
), NULL
, 0, &needed
);
870 ok( res
== ERROR_SUCCESS
, "returned %d with %u (expected ERROR_SUCCESS)\n", res
, GetLastError());
871 needed
= sizeof(buffer
)-1 ;
872 res
= RegQueryValueExA(hroot
, TransmissionRetryTimeoutA
, NULL
, NULL
, (PBYTE
) buffer
, &needed
);
873 ok( (res
== ERROR_SUCCESS
) && (lstrcmpA(buffer
, num_1A
) == 0),
874 "returned %d and '%s' (expected ERROR_SUCCESS and '%s')\n",
875 res
, buffer
, num_1A
);
877 /* set to "999999" */
878 needed
= (DWORD
) 0xdeadbeef;
879 SetLastError(0xdeadbeef);
880 res
= pXcvDataPort(hXcv
, cmd_ConfigureLPTPortCommandOKW
, (PBYTE
) num_999999W
, sizeof(num_999999W
), NULL
, 0, &needed
);
881 ok( res
== ERROR_SUCCESS
, "returned %d with %u (expected ERROR_SUCCESS)\n", res
, GetLastError());
882 needed
= sizeof(buffer
)-1 ;
883 res
= RegQueryValueExA(hroot
, TransmissionRetryTimeoutA
, NULL
, NULL
, (PBYTE
) buffer
, &needed
);
884 ok( (res
== ERROR_SUCCESS
) && (lstrcmpA(buffer
, num_999999A
) == 0),
885 "returned %d and '%s' (expected ERROR_SUCCESS and '%s')\n",
886 res
, buffer
, num_999999A
);
888 /* set to "1000000" */
889 needed
= (DWORD
) 0xdeadbeef;
890 SetLastError(0xdeadbeef);
891 res
= pXcvDataPort(hXcv
, cmd_ConfigureLPTPortCommandOKW
, (PBYTE
) num_1000000W
, sizeof(num_1000000W
), NULL
, 0, &needed
);
892 ok( res
== ERROR_SUCCESS
, "returned %d with %u (expected ERROR_SUCCESS)\n", res
, GetLastError());
893 needed
= sizeof(buffer
)-1 ;
894 res
= RegQueryValueExA(hroot
, TransmissionRetryTimeoutA
, NULL
, NULL
, (PBYTE
) buffer
, &needed
);
895 ok( (res
== ERROR_SUCCESS
) && (lstrcmpA(buffer
, num_1000000A
) == 0),
896 "returned %d and '%s' (expected ERROR_SUCCESS and '%s')\n",
897 res
, buffer
, num_1000000A
);
899 /* using cmd_ConfigureLPTPortCommandOKW with does_not_existW:
900 the string "does_not_exist" is written to the registry */
903 /* restore the original value */
904 RegDeleteValueA(hroot
, TransmissionRetryTimeoutA
);
906 res
= RegSetValueExA(hroot
, TransmissionRetryTimeoutA
, 0, REG_SZ
, (PBYTE
)org_value
, lstrlenA(org_value
)+1);
907 ok(res
== ERROR_SUCCESS
, "unable to restore original value (got %u): %s\n", res
, org_value
);
914 /* ########################### */
916 static void test_XcvDataPort_DeletePort(void)
922 /* cleanup: just to make sure */
923 needed
= (DWORD
) 0xdeadbeef;
924 SetLastError(0xdeadbeef);
925 res
= pXcvDataPort(hXcv
, cmd_DeletePortW
, (PBYTE
) tempfileW
, (lstrlenW(tempfileW
) + 1) * sizeof(WCHAR
), NULL
, 0, &needed
);
926 ok( !res
|| (res
== ERROR_FILE_NOT_FOUND
),
927 "returned %d with %u (expected ERROR_SUCCESS or ERROR_FILE_NOT_FOUND)\n",
928 res
, GetLastError());
931 /* ToDo: cmd_AddPortW for tempfileW, then cmd_DeletePortW for the existing Port */
934 /* try to delete a nonexistent Port */
935 needed
= (DWORD
) 0xdeadbeef;
936 SetLastError(0xdeadbeef);
937 res
= pXcvDataPort(hXcv
, cmd_DeletePortW
, (PBYTE
) tempfileW
, (lstrlenW(tempfileW
) + 1) * sizeof(WCHAR
), NULL
, 0, &needed
);
938 ok( res
== ERROR_FILE_NOT_FOUND
,
939 "returned %d with %u (expected ERROR_FILE_NOT_FOUND)\n", res
, GetLastError());
941 /* emptyW as Portname: ERROR_FILE_NOT_FOUND is returned */
942 /* NULL as Portname: Native localspl.dll crashed */
946 /* ########################### */
948 static void test_XcvDataPort_GetTransmissionRetryTimeout(void)
958 /* ask for needed size */
959 needed
= (DWORD
) 0xdeadbeef;
960 SetLastError(0xdeadbeef);
961 res
= pXcvDataPort(hXcv
, cmd_GetTransmissionRetryTimeoutW
, NULL
, 0, NULL
, 0, &needed
);
962 if (res
== ERROR_INVALID_PARAMETER
) {
963 skip("'GetTransmissionRetryTimeout' not supported\n");
967 ok( (res
== ERROR_INSUFFICIENT_BUFFER
) && (needed
== len
),
968 "returned %d with %u and %u (expected ERROR_INSUFFICIENT_BUFFER "
969 "and '%u')\n", res
, GetLastError(), needed
, len
);
972 /* Read the original value from the registry */
973 res
= RegOpenKeyExA(HKEY_LOCAL_MACHINE
, WinNT_CV_WindowsA
, 0, KEY_ALL_ACCESS
, &hroot
);
974 if (res
== ERROR_ACCESS_DENIED
) {
975 skip("ACCESS_DENIED\n");
979 if (res
!= ERROR_SUCCESS
) {
980 /* unable to open the registry: skip the test */
981 skip("got %d\n", res
);
986 needed
= sizeof(org_value
)-1 ;
987 res
= RegQueryValueExA(hroot
, TransmissionRetryTimeoutA
, NULL
, NULL
, (PBYTE
) org_value
, &needed
);
988 ok( (res
== ERROR_SUCCESS
) || (res
== ERROR_FILE_NOT_FOUND
),
989 "returned %u and %u for \"%s\" (expected ERROR_SUCCESS or "
990 "ERROR_FILE_NOT_FOUND)\n", res
, needed
, org_value
);
992 /* Get default value (documented as 90 in the w2k reskit, but that is wrong) */
993 RegDeleteValueA(hroot
, TransmissionRetryTimeoutA
);
994 needed
= (DWORD
) 0xdeadbeef;
995 buffer
[0] = 0xdeadbeef;
996 SetLastError(0xdeadbeef);
997 res
= pXcvDataPort(hXcv
, cmd_GetTransmissionRetryTimeoutW
, NULL
, 0, (PBYTE
) buffer
, len
, &needed
);
998 ok( (res
== ERROR_SUCCESS
) && (buffer
[0] == 45),
999 "returned %d with %u and %u for %d\n (expected ERROR_SUCCESS "
1000 "for '45')\n", res
, GetLastError(), needed
, buffer
[0]);
1002 /* the default timeout is returned, when the value is empty */
1003 res
= RegSetValueExA(hroot
, TransmissionRetryTimeoutA
, 0, REG_SZ
, (PBYTE
)emptyA
, 1);
1004 needed
= (DWORD
) 0xdeadbeef;
1005 buffer
[0] = 0xdeadbeef;
1006 SetLastError(0xdeadbeef);
1007 res
= pXcvDataPort(hXcv
, cmd_GetTransmissionRetryTimeoutW
, NULL
, 0, (PBYTE
) buffer
, len
, &needed
);
1008 ok( (res
== ERROR_SUCCESS
) && (buffer
[0] == 45),
1009 "returned %d with %u and %u for %d\n (expected ERROR_SUCCESS "
1010 "for '45')\n", res
, GetLastError(), needed
, buffer
[0]);
1012 /* the dialog is limited (1 - 999999), but that is done somewhere else */
1013 res
= RegSetValueExA(hroot
, TransmissionRetryTimeoutA
, 0, REG_SZ
, (PBYTE
)num_0A
, lstrlenA(num_0A
)+1);
1014 needed
= (DWORD
) 0xdeadbeef;
1015 buffer
[0] = 0xdeadbeef;
1016 SetLastError(0xdeadbeef);
1017 res
= pXcvDataPort(hXcv
, cmd_GetTransmissionRetryTimeoutW
, NULL
, 0, (PBYTE
) buffer
, len
, &needed
);
1018 ok( (res
== ERROR_SUCCESS
) && (buffer
[0] == 0),
1019 "returned %d with %u and %u for %d\n (expected ERROR_SUCCESS "
1020 "for '0')\n", res
, GetLastError(), needed
, buffer
[0]);
1023 res
= RegSetValueExA(hroot
, TransmissionRetryTimeoutA
, 0, REG_SZ
, (PBYTE
)num_1A
, lstrlenA(num_1A
)+1);
1024 needed
= (DWORD
) 0xdeadbeef;
1025 buffer
[0] = 0xdeadbeef;
1026 SetLastError(0xdeadbeef);
1027 res
= pXcvDataPort(hXcv
, cmd_GetTransmissionRetryTimeoutW
, NULL
, 0, (PBYTE
) buffer
, len
, &needed
);
1028 ok( (res
== ERROR_SUCCESS
) && (buffer
[0] == 1),
1029 "returned %d with %u and %u for %d\n (expected 'ERROR_SUCCESS' "
1030 "for '1')\n", res
, GetLastError(), needed
, buffer
[0]);
1032 res
= RegSetValueExA(hroot
, TransmissionRetryTimeoutA
, 0, REG_SZ
, (PBYTE
)num_999999A
, lstrlenA(num_999999A
)+1);
1033 needed
= (DWORD
) 0xdeadbeef;
1034 buffer
[0] = 0xdeadbeef;
1035 SetLastError(0xdeadbeef);
1036 res
= pXcvDataPort(hXcv
, cmd_GetTransmissionRetryTimeoutW
, NULL
, 0, (PBYTE
) buffer
, len
, &needed
);
1037 ok( (res
== ERROR_SUCCESS
) && (buffer
[0] == 999999),
1038 "returned %d with %u and %u for %d\n (expected ERROR_SUCCESS "
1039 "for '999999')\n", res
, GetLastError(), needed
, buffer
[0]);
1042 res
= RegSetValueExA(hroot
, TransmissionRetryTimeoutA
, 0, REG_SZ
, (PBYTE
)num_1000000A
, lstrlenA(num_1000000A
)+1);
1043 needed
= (DWORD
) 0xdeadbeef;
1044 buffer
[0] = 0xdeadbeef;
1045 SetLastError(0xdeadbeef);
1046 res
= pXcvDataPort(hXcv
, cmd_GetTransmissionRetryTimeoutW
, NULL
, 0, (PBYTE
) buffer
, len
, &needed
);
1047 ok( (res
== ERROR_SUCCESS
) && (buffer
[0] == 1000000),
1048 "returned %d with %u and %u for %d\n (expected ERROR_SUCCESS "
1049 "for '1000000')\n", res
, GetLastError(), needed
, buffer
[0]);
1051 /* restore the original value */
1052 RegDeleteValueA(hroot
, TransmissionRetryTimeoutA
);
1054 res
= RegSetValueExA(hroot
, TransmissionRetryTimeoutA
, 0, REG_SZ
, (PBYTE
)org_value
, lstrlenA(org_value
)+1);
1055 ok(res
== ERROR_SUCCESS
, "unable to restore original value (got %u): %s\n", res
, org_value
);
1061 /* ########################### */
1063 static void test_XcvDataPort_MonitorUI(void)
1066 BYTE buffer
[MAX_PATH
+ 2];
1071 /* ask for needed size */
1072 needed
= (DWORD
) 0xdeadbeef;
1073 SetLastError(0xdeadbeef);
1074 res
= pXcvDataPort(hXcv
, cmd_MonitorUIW
, NULL
, 0, NULL
, 0, &needed
);
1075 if (res
== ERROR_INVALID_PARAMETER
) {
1076 skip("'MonitorUI' nor supported\n");
1079 ok( (res
== ERROR_INSUFFICIENT_BUFFER
) && (needed
<= MAX_PATH
),
1080 "returned %d with %u and 0x%x (expected 'ERROR_INSUFFICIENT_BUFFER' "
1081 " and '<= MAX_PATH')\n", res
, GetLastError(), needed
);
1083 if (needed
> MAX_PATH
) {
1084 skip("buffer overflow (%u)\n", needed
);
1089 /* the command is required */
1090 needed
= (DWORD
) 0xdeadbeef;
1091 SetLastError(0xdeadbeef);
1092 res
= pXcvDataPort(hXcv
, emptyW
, NULL
, 0, NULL
, 0, &needed
);
1093 ok( res
== ERROR_INVALID_PARAMETER
, "returned %d with %u and 0x%x "
1094 "(expected 'ERROR_INVALID_PARAMETER')\n", res
, GetLastError(), needed
);
1097 /* crash with native localspl.dll (w2k+xp) */
1098 res
= pXcvDataPort(hXcv
, NULL
, NULL
, 0, buffer
, MAX_PATH
, &needed
);
1099 res
= pXcvDataPort(hXcv
, cmd_MonitorUIW
, NULL
, 0, NULL
, len
, &needed
);
1100 res
= pXcvDataPort(hXcv
, cmd_MonitorUIW
, NULL
, 0, buffer
, len
, NULL
);
1104 /* hXcv is ignored for the command "MonitorUI" */
1105 needed
= (DWORD
) 0xdeadbeef;
1106 SetLastError(0xdeadbeef);
1107 res
= pXcvDataPort(NULL
, cmd_MonitorUIW
, NULL
, 0, buffer
, len
, &needed
);
1108 ok( res
== ERROR_SUCCESS
, "returned %d with %u and 0x%x "
1109 "(expected 'ERROR_SUCCESS')\n", res
, GetLastError(), needed
);
1112 /* pszDataName is case-sensitive */
1113 memset(buffer
, 0, len
);
1114 needed
= (DWORD
) 0xdeadbeef;
1115 SetLastError(0xdeadbeef);
1116 res
= pXcvDataPort(hXcv
, cmd_MonitorUI_lcaseW
, NULL
, 0, buffer
, len
, &needed
);
1117 ok( res
== ERROR_INVALID_PARAMETER
, "returned %d with %u and 0x%x "
1118 "(expected 'ERROR_INVALID_PARAMETER')\n", res
, GetLastError(), needed
);
1120 /* off by one: larger */
1121 needed
= (DWORD
) 0xdeadbeef;
1122 SetLastError(0xdeadbeef);
1123 res
= pXcvDataPort(hXcv
, cmd_MonitorUIW
, NULL
, 0, buffer
, len
+1, &needed
);
1124 ok( res
== ERROR_SUCCESS
, "returned %d with %u and 0x%x "
1125 "(expected 'ERROR_SUCCESS')\n", res
, GetLastError(), needed
);
1128 /* off by one: smaller */
1129 /* the buffer is not modified for NT4, w2k, XP */
1130 needed
= (DWORD
) 0xdeadbeef;
1131 SetLastError(0xdeadbeef);
1132 res
= pXcvDataPort(hXcv
, cmd_MonitorUIW
, NULL
, 0, buffer
, len
-1, &needed
);
1133 ok( res
== ERROR_INSUFFICIENT_BUFFER
, "returned %d with %u and 0x%x "
1134 "(expected 'ERROR_INSUFFICIENT_BUFFER')\n", res
, GetLastError(), needed
);
1136 /* Normal use. The DLL-Name without a Path is returned */
1137 memset(buffer
, 0, len
);
1138 needed
= (DWORD
) 0xdeadbeef;
1139 SetLastError(0xdeadbeef);
1140 res
= pXcvDataPort(hXcv
, cmd_MonitorUIW
, NULL
, 0, buffer
, len
, &needed
);
1141 ok( res
== ERROR_SUCCESS
, "returned %d with %u and 0x%x "
1142 "(expected 'ERROR_SUCCESS')\n", res
, GetLastError(), needed
);
1145 /* small check without access-rights: */
1146 if (!hXcv_noaccess
) return;
1148 /* The ACCESS_MASK is ignored for "MonitorUI" */
1149 memset(buffer
, 0, len
);
1150 needed
= (DWORD
) 0xdeadbeef;
1151 SetLastError(0xdeadbeef);
1152 res
= pXcvDataPort(hXcv_noaccess
, cmd_MonitorUIW
, NULL
, 0, buffer
, sizeof(buffer
), &needed
);
1153 ok( res
== ERROR_SUCCESS
, "returned %d with %u and 0x%x "
1154 "(expected 'ERROR_SUCCESS')\n", res
, GetLastError(), needed
);
1157 /* ########################### */
1159 static void test_XcvDataPort_PortIsValid(void)
1164 /* normal use: "LPT1:" */
1165 needed
= (DWORD
) 0xdeadbeef;
1166 SetLastError(0xdeadbeef);
1167 res
= pXcvDataPort(hXcv
, cmd_PortIsValidW
, (PBYTE
) portname_lpt1W
, sizeof(portname_lpt1W
), NULL
, 0, &needed
);
1168 if (res
== ERROR_INVALID_PARAMETER
) {
1169 skip("'PostIsValid' not supported\n");
1172 ok( res
== ERROR_SUCCESS
, "returned %d with %u (expected ERROR_SUCCESS)\n", res
, GetLastError());
1176 /* crash with native localspl.dll (w2k+xp) */
1177 res
= pXcvDataPort(hXcv
, cmd_PortIsValidW
, NULL
, 0, NULL
, 0, &needed
);
1181 /* hXcv is ignored for the command "PortIsValid" */
1182 needed
= (DWORD
) 0xdeadbeef;
1183 SetLastError(0xdeadbeef);
1184 res
= pXcvDataPort(NULL
, cmd_PortIsValidW
, (PBYTE
) portname_lpt1W
, sizeof(portname_lpt1W
), NULL
, 0, NULL
);
1185 ok( res
== ERROR_SUCCESS
, "returned %d with %u (expected ERROR_SUCCESS)\n", res
, GetLastError());
1187 /* needed is ignored */
1188 needed
= (DWORD
) 0xdeadbeef;
1189 SetLastError(0xdeadbeef);
1190 res
= pXcvDataPort(hXcv
, cmd_PortIsValidW
, (PBYTE
) portname_lpt1W
, sizeof(portname_lpt1W
), NULL
, 0, NULL
);
1191 ok( res
== ERROR_SUCCESS
, "returned %d with %u (expected ERROR_SUCCESS)\n", res
, GetLastError());
1194 /* cbInputData is ignored */
1195 needed
= (DWORD
) 0xdeadbeef;
1196 SetLastError(0xdeadbeef);
1197 res
= pXcvDataPort(hXcv
, cmd_PortIsValidW
, (PBYTE
) portname_lpt1W
, 0, NULL
, 0, &needed
);
1198 ok( res
== ERROR_SUCCESS
,
1199 "returned %d with %u and 0x%x (expected ERROR_SUCCESS)\n",
1200 res
, GetLastError(), needed
);
1202 needed
= (DWORD
) 0xdeadbeef;
1203 SetLastError(0xdeadbeef);
1204 res
= pXcvDataPort(hXcv
, cmd_PortIsValidW
, (PBYTE
) portname_lpt1W
, 1, NULL
, 0, &needed
);
1205 ok( res
== ERROR_SUCCESS
,
1206 "returned %d with %u and 0x%x (expected ERROR_SUCCESS)\n",
1207 res
, GetLastError(), needed
);
1209 needed
= (DWORD
) 0xdeadbeef;
1210 SetLastError(0xdeadbeef);
1211 res
= pXcvDataPort(hXcv
, cmd_PortIsValidW
, (PBYTE
) portname_lpt1W
, sizeof(portname_lpt1W
) -1, NULL
, 0, &needed
);
1212 ok( res
== ERROR_SUCCESS
,
1213 "returned %d with %u and 0x%x (expected ERROR_SUCCESS)\n",
1214 res
, GetLastError(), needed
);
1216 needed
= (DWORD
) 0xdeadbeef;
1217 SetLastError(0xdeadbeef);
1218 res
= pXcvDataPort(hXcv
, cmd_PortIsValidW
, (PBYTE
) portname_lpt1W
, sizeof(portname_lpt1W
) -2, NULL
, 0, &needed
);
1219 ok( res
== ERROR_SUCCESS
,
1220 "returned %d with %u and 0x%x (expected ERROR_SUCCESS)\n",
1221 res
, GetLastError(), needed
);
1224 /* an empty name is not allowed */
1225 needed
= (DWORD
) 0xdeadbeef;
1226 SetLastError(0xdeadbeef);
1227 res
= pXcvDataPort(hXcv
, cmd_PortIsValidW
, (PBYTE
) emptyW
, sizeof(emptyW
), NULL
, 0, &needed
);
1228 ok( res
== ERROR_PATH_NOT_FOUND
,
1229 "returned %d with %u and 0x%x (expected ERROR_PATH_NOT_FOUND)\n",
1230 res
, GetLastError(), needed
);
1233 /* a directory is not allowed */
1234 needed
= (DWORD
) 0xdeadbeef;
1235 SetLastError(0xdeadbeef);
1236 res
= pXcvDataPort(hXcv
, cmd_PortIsValidW
, (PBYTE
) tempdirW
, (lstrlenW(tempdirW
) + 1) * sizeof(WCHAR
), NULL
, 0, &needed
);
1237 /* XP(admin): ERROR_INVALID_NAME, XP(user): ERROR_PATH_NOT_FOUND, w2k ERROR_ACCESS_DENIED */
1238 ok( (res
== ERROR_INVALID_NAME
) || (res
== ERROR_PATH_NOT_FOUND
) ||
1239 (res
== ERROR_ACCESS_DENIED
), "returned %d with %u and 0x%x "
1240 "(expected ERROR_INVALID_NAME, ERROR_PATH_NOT_FOUND or ERROR_ACCESS_DENIED)\n",
1241 res
, GetLastError(), needed
);
1244 /* test more valid well known Ports: */
1245 needed
= (DWORD
) 0xdeadbeef;
1246 SetLastError(0xdeadbeef);
1247 res
= pXcvDataPort(hXcv
, cmd_PortIsValidW
, (PBYTE
) portname_lpt2W
, sizeof(portname_lpt2W
), NULL
, 0, &needed
);
1248 ok( res
== ERROR_SUCCESS
,
1249 "returned %d with %u and 0x%x (expected ERROR_SUCCESS)\n",
1250 res
, GetLastError(), needed
);
1253 needed
= (DWORD
) 0xdeadbeef;
1254 SetLastError(0xdeadbeef);
1255 res
= pXcvDataPort(hXcv
, cmd_PortIsValidW
, (PBYTE
) portname_com1W
, sizeof(portname_com1W
), NULL
, 0, &needed
);
1256 ok( res
== ERROR_SUCCESS
,
1257 "returned %d with %u and 0x%x (expected ERROR_SUCCESS)\n",
1258 res
, GetLastError(), needed
);
1261 needed
= (DWORD
) 0xdeadbeef;
1262 SetLastError(0xdeadbeef);
1263 res
= pXcvDataPort(hXcv
, cmd_PortIsValidW
, (PBYTE
) portname_com2W
, sizeof(portname_com2W
), NULL
, 0, &needed
);
1264 ok( res
== ERROR_SUCCESS
,
1265 "returned %d with %u and 0x%x (expected ERROR_SUCCESS)\n",
1266 res
, GetLastError(), needed
);
1269 needed
= (DWORD
) 0xdeadbeef;
1270 SetLastError(0xdeadbeef);
1271 res
= pXcvDataPort(hXcv
, cmd_PortIsValidW
, (PBYTE
) portname_fileW
, sizeof(portname_fileW
), NULL
, 0, &needed
);
1272 ok( res
== ERROR_SUCCESS
,
1273 "returned %d with %u and 0x%x (expected ERROR_SUCCESS)\n",
1274 res
, GetLastError(), needed
);
1277 /* a normal, writable file is allowed */
1278 needed
= (DWORD
) 0xdeadbeef;
1279 SetLastError(0xdeadbeef);
1280 res
= pXcvDataPort(hXcv
, cmd_PortIsValidW
, (PBYTE
) tempfileW
, (lstrlenW(tempfileW
) + 1) * sizeof(WCHAR
), NULL
, 0, &needed
);
1281 ok( res
== ERROR_SUCCESS
,
1282 "returned %d with %u and 0x%x (expected ERROR_SUCCESS)\n",
1283 res
, GetLastError(), needed
);
1286 /* small check without access-rights: */
1287 if (!hXcv_noaccess
) return;
1289 /* The ACCESS_MASK from XcvOpenPort is ignored in "PortIsValid" */
1290 needed
= (DWORD
) 0xdeadbeef;
1291 SetLastError(0xdeadbeef);
1292 res
= pXcvDataPort(hXcv_noaccess
, cmd_PortIsValidW
, (PBYTE
) portname_lpt1W
, sizeof(portname_lpt1W
), NULL
, 0, &needed
);
1293 ok( res
== ERROR_SUCCESS
, "returned %d with %u (expected ERROR_SUCCESS)\n", res
, GetLastError());
1297 /* ########################### */
1299 static void test_XcvOpenPort(void)
1307 /* crash with native localspl.dll (w2k+xp) */
1308 res
= pXcvOpenPort(NULL
, SERVER_ACCESS_ADMINISTER
, &hXcv2
);
1309 res
= pXcvOpenPort(emptyW
, SERVER_ACCESS_ADMINISTER
, NULL
);
1313 /* The returned handle is the result from a previous "spoolss.dll,DllAllocSplMem" */
1314 SetLastError(0xdeadbeef);
1315 hXcv2
= (HANDLE
) 0xdeadbeef;
1316 res
= pXcvOpenPort(emptyW
, SERVER_ACCESS_ADMINISTER
, &hXcv2
);
1317 ok(res
, "returned %d with %u and %p (expected '!= 0')\n", res
, GetLastError(), hXcv2
);
1318 if (res
) pXcvClosePort(hXcv2
);
1321 /* The ACCESS_MASK is not checked in XcvOpenPort */
1322 SetLastError(0xdeadbeef);
1323 hXcv2
= (HANDLE
) 0xdeadbeef;
1324 res
= pXcvOpenPort(emptyW
, 0, &hXcv2
);
1325 ok(res
, "returned %d with %u and %p (expected '!= 0')\n", res
, GetLastError(), hXcv2
);
1326 if (res
) pXcvClosePort(hXcv2
);
1329 /* A copy of pszObject is saved in the Memory-Block */
1330 SetLastError(0xdeadbeef);
1331 hXcv2
= (HANDLE
) 0xdeadbeef;
1332 res
= pXcvOpenPort(portname_lpt1W
, SERVER_ALL_ACCESS
, &hXcv2
);
1333 ok(res
, "returned %d with %u and %p (expected '!= 0')\n", res
, GetLastError(), hXcv2
);
1334 if (res
) pXcvClosePort(hXcv2
);
1336 SetLastError(0xdeadbeef);
1337 hXcv2
= (HANDLE
) 0xdeadbeef;
1338 res
= pXcvOpenPort(portname_fileW
, SERVER_ALL_ACCESS
, &hXcv2
);
1339 ok(res
, "returned %d with %u and %p (expected '!= 0')\n", res
, GetLastError(), hXcv2
);
1340 if (res
) pXcvClosePort(hXcv2
);
1344 /* ########################### */
1346 #define GET_MONITOR_FUNC(name) \
1347 if(numentries > 0) { \
1349 p##name = pm->Monitor.pfn##name ; \
1353 START_TEST(localmon
)
1358 LoadLibraryA("winspool.drv");
1359 /* This DLL does not exist on Win9x */
1360 hdll
= LoadLibraryA("localspl.dll");
1362 skip("localspl.dll cannot be loaded, most likely running on Win9x\n");
1367 tempfileW
[0] = '\0';
1368 res
= GetTempPathW(MAX_PATH
, tempdirW
);
1369 ok(res
!= 0, "with %u\n", GetLastError());
1370 res
= GetTempFileNameW(tempdirW
, wineW
, 0, tempfileW
);
1371 ok(res
!= 0, "with %u\n", GetLastError());
1373 pInitializePrintMonitor
= (void *) GetProcAddress(hdll
, "InitializePrintMonitor");
1375 if (!pInitializePrintMonitor
) {
1376 /* The Monitor for "Local Ports" was in a separate dll before w2k */
1377 hlocalmon
= LoadLibraryA("localmon.dll");
1379 pInitializePrintMonitor
= (void *) GetProcAddress(hlocalmon
, "InitializePrintMonitor");
1382 if (!pInitializePrintMonitor
) return;
1384 /* Native localmon.dll / localspl.dll need a valid Port-Entry in:
1385 a) since xp: HKLM\Software\Microsoft\Windows NT\CurrentVersion\Ports
1386 b) up to w2k: Section "Ports" in win.ini
1387 or InitializePrintMonitor fails. */
1388 pm
= pInitializePrintMonitor(Monitors_LocalPortW
);
1390 numentries
= (pm
->dwMonitorSize
) / sizeof(VOID
*);
1391 /* NT4: 14, since w2k: 17 */
1392 ok( numentries
== 14 || numentries
== 17,
1393 "dwMonitorSize (%d) => %d Functions\n", pm
->dwMonitorSize
, numentries
);
1395 GET_MONITOR_FUNC(EnumPorts
);
1396 GET_MONITOR_FUNC(OpenPort
);
1397 GET_MONITOR_FUNC(OpenPortEx
);
1398 GET_MONITOR_FUNC(StartDocPort
);
1399 GET_MONITOR_FUNC(WritePort
);
1400 GET_MONITOR_FUNC(ReadPort
);
1401 GET_MONITOR_FUNC(EndDocPort
);
1402 GET_MONITOR_FUNC(ClosePort
);
1403 GET_MONITOR_FUNC(AddPort
);
1404 GET_MONITOR_FUNC(AddPortEx
);
1405 GET_MONITOR_FUNC(ConfigurePort
);
1406 GET_MONITOR_FUNC(DeletePort
);
1407 GET_MONITOR_FUNC(GetPrinterDataFromPort
);
1408 GET_MONITOR_FUNC(SetPortTimeOuts
);
1409 GET_MONITOR_FUNC(XcvOpenPort
);
1410 GET_MONITOR_FUNC(XcvDataPort
);
1411 GET_MONITOR_FUNC(XcvClosePort
);
1413 if ((pXcvOpenPort
) && (pXcvDataPort
) && (pXcvClosePort
)) {
1414 SetLastError(0xdeadbeef);
1415 res
= pXcvOpenPort(emptyW
, SERVER_ACCESS_ADMINISTER
, &hXcv
);
1416 ok(res
, "hXcv: %d with %u and %p (expected '!= 0')\n", res
, GetLastError(), hXcv
);
1418 SetLastError(0xdeadbeef);
1419 res
= pXcvOpenPort(emptyW
, 0, &hXcv_noaccess
);
1420 ok(res
, "hXcv_noaccess: %d with %u and %p (expected '!= 0')\n", res
, GetLastError(), hXcv_noaccess
);
1424 test_InitializePrintMonitor();
1426 find_installed_ports();
1431 test_ConfigurePort();
1437 skip("Xcv not supported\n");
1441 test_XcvClosePort();
1442 test_XcvDataPort_AddPort();
1443 test_XcvDataPort_ConfigureLPTPortCommandOK();
1444 test_XcvDataPort_DeletePort();
1445 test_XcvDataPort_GetTransmissionRetryTimeout();
1446 test_XcvDataPort_MonitorUI();
1447 test_XcvDataPort_PortIsValid();
1450 pXcvClosePort(hXcv
);
1452 if (hXcv_noaccess
) pXcvClosePort(hXcv_noaccess
);
1454 /* Cleanup our temporary file */
1455 DeleteFileW(tempfileW
);