2 * Unit tests for console API
4 * Copyright (c) 2003,2004 Eric Pouech
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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
21 #include "wine/test.h"
25 /* DEFAULT_ATTRIB is used for all initial filling of the console.
26 * all modifications are made with TEST_ATTRIB so that we could check
27 * what has to be modified or not
29 #define TEST_ATTRIB (BACKGROUND_BLUE | FOREGROUND_GREEN)
30 #define DEFAULT_ATTRIB (FOREGROUND_GREEN | FOREGROUND_BLUE | FOREGROUND_RED)
31 /* when filling the screen with non-blank chars, this macro defines
32 * what character should be at position 'c'
34 #define CONTENT(c) ('A' + (((c).Y * 17 + (c).X) % 23))
36 #define okCURSOR(hCon, c) do { \
37 CONSOLE_SCREEN_BUFFER_INFO __sbi; \
38 BOOL expect = GetConsoleScreenBufferInfo((hCon), &__sbi) && \
39 __sbi.dwCursorPosition.X == (c).X && __sbi.dwCursorPosition.Y == (c).Y; \
40 ok(expect, "Expected cursor at (%d,%d), got (%d,%d)\n", \
41 (c).X, (c).Y, __sbi.dwCursorPosition.X, __sbi.dwCursorPosition.Y); \
44 #define okCHAR(hCon, c, ch, attr) do { \
45 char __ch; WORD __attr; DWORD __len; BOOL expect; \
46 expect = ReadConsoleOutputCharacter((hCon), &__ch, 1, (c), &__len) == 1 && __len == 1 && __ch == (ch); \
47 ok(expect, "At (%d,%d): expecting char '%c'/%02x got '%c'/%02x\n", (c).X, (c).Y, (ch), (ch), __ch, __ch); \
48 expect = ReadConsoleOutputAttribute((hCon), &__attr, 1, (c), &__len) == 1 && __len == 1 && __attr == (attr); \
49 ok(expect, "At (%d,%d): expecting attr %04x got %04x\n", (c).X, (c).Y, (attr), __attr); \
52 /* FIXME: this could be optimized on a speed point of view */
53 static void resetContent(HANDLE hCon
, COORD sbSize
, BOOL content
)
56 WORD attr
= DEFAULT_ATTRIB
;
60 for (c
.X
= 0; c
.X
< sbSize
.X
; c
.X
++)
62 for (c
.Y
= 0; c
.Y
< sbSize
.Y
; c
.Y
++)
64 ch
= (content
) ? CONTENT(c
) : ' ';
65 WriteConsoleOutputAttribute(hCon
, &attr
, 1, c
, &len
);
66 WriteConsoleOutputCharacterA(hCon
, &ch
, 1, c
, &len
);
71 static void testCursor(HANDLE hCon
, COORD sbSize
)
76 ok(SetConsoleCursorPosition(0, c
) == 0, "No handle\n");
77 ok(GetLastError() == ERROR_INVALID_HANDLE
, "GetLastError: expecting %u got %lu\n",
78 ERROR_INVALID_HANDLE
, GetLastError());
81 ok(SetConsoleCursorPosition(hCon
, c
) != 0, "Cursor in upper-left\n");
86 ok(SetConsoleCursorPosition(hCon
, c
) != 0, "Cursor in lower-right\n");
91 ok(SetConsoleCursorPosition(hCon
, c
) == 0, "Cursor is outside\n");
92 ok(GetLastError() == ERROR_INVALID_PARAMETER
, "GetLastError: expecting %u got %lu\n",
93 ERROR_INVALID_PARAMETER
, GetLastError());
97 ok(SetConsoleCursorPosition(hCon
, c
) == 0, "Cursor is outside\n");
98 ok(GetLastError() == ERROR_INVALID_PARAMETER
, "GetLastError: expecting %u got %lu\n",
99 ERROR_INVALID_PARAMETER
, GetLastError());
103 ok(SetConsoleCursorPosition(hCon
, c
) == 0, "Cursor is outside\n");
104 ok(GetLastError() == ERROR_INVALID_PARAMETER
, "GetLastError: expecting %u got %lu\n",
105 ERROR_INVALID_PARAMETER
, GetLastError());
109 ok(SetConsoleCursorPosition(hCon
, c
) == 0, "Cursor is outside\n");
110 ok(GetLastError() == ERROR_INVALID_PARAMETER
, "GetLastError: expecting %u got %lu\n",
111 ERROR_INVALID_PARAMETER
, GetLastError());
114 static void testWriteSimple(HANDLE hCon
, COORD sbSize
)
118 const char* mytest
= "abcdefg";
119 const int mylen
= strlen(mytest
);
121 /* single line write */
123 ok(SetConsoleCursorPosition(hCon
, c
) != 0, "Cursor in upper-left\n");
125 ok(WriteConsole(hCon
, mytest
, mylen
, &len
, NULL
) != 0 && len
== mylen
, "WriteConsole\n");
127 for (c
.X
= 0; c
.X
< mylen
; c
.X
++)
129 okCHAR(hCon
, c
, mytest
[c
.X
], TEST_ATTRIB
);
133 okCHAR(hCon
, c
, ' ', DEFAULT_ATTRIB
);
136 static void testWriteNotWrappedNotProcessed(HANDLE hCon
, COORD sbSize
)
145 ok(GetConsoleMode(hCon
, &mode
) && SetConsoleMode(hCon
, mode
& ~(ENABLE_PROCESSED_OUTPUT
|ENABLE_WRAP_AT_EOL_OUTPUT
)),
146 "clearing wrap at EOL & processed output\n");
148 /* write line, wrapping disabled, buffer exceeds sb width */
149 c
.X
= sbSize
.X
- 3; c
.Y
= 0;
150 ok(SetConsoleCursorPosition(hCon
, c
) != 0, "Cursor in upper-left-3\n");
154 mylen
= strlen(mytest
);
156 ret
= WriteConsole(hCon
, mytest
, mylen
, &len
, NULL
);
157 ok(ret
!= 0 && len
== mylen
, "Couldn't write, ret = %d, len = %ld\n", ret
, len
);
159 for (p
= mylen
- 3; p
< mylen
; p
++)
161 c
.X
= sbSize
.X
- 3 + p
% 3;
162 okCHAR(hCon
, c
, mytest
[p
], TEST_ATTRIB
);
166 okCHAR(hCon
, c
, ' ', DEFAULT_ATTRIB
);
168 p
= sbSize
.X
- 3 + mylen
% 3;
171 /* write line, wrapping disabled, strings end on end of line */
172 c
.X
= sbSize
.X
- mylen
; c
.Y
= 0;
173 ok(SetConsoleCursorPosition(hCon
, c
) != 0, "Cursor in upper-left-3\n");
175 ok(WriteConsole(hCon
, mytest
, mylen
, &len
, NULL
) != 0 && len
== mylen
, "WriteConsole\n");
178 static void testWriteNotWrappedProcessed(HANDLE hCon
, COORD sbSize
)
182 const char* mytest
= "abcd\nf\tg";
183 const int mylen
= strlen(mytest
);
184 const int mylen2
= strchr(mytest
, '\n') - mytest
;
187 ok(GetConsoleMode(hCon
, &mode
) && SetConsoleMode(hCon
, (mode
| ENABLE_PROCESSED_OUTPUT
) & ~ENABLE_WRAP_AT_EOL_OUTPUT
),
188 "clearing wrap at EOL & setting processed output\n");
190 /* write line, wrapping disabled, buffer exceeds sb width */
191 c
.X
= sbSize
.X
- 5; c
.Y
= 0;
192 ok(SetConsoleCursorPosition(hCon
, c
) != 0, "Cursor in upper-left-5\n");
194 ok(WriteConsole(hCon
, mytest
, mylen
, &len
, NULL
) != 0 && len
== mylen
, "WriteConsole\n");
196 for (c
.X
= sbSize
.X
- 5; c
.X
< sbSize
.X
- 1; c
.X
++)
198 okCHAR(hCon
, c
, mytest
[c
.X
- sbSize
.X
+ 5], TEST_ATTRIB
);
200 okCHAR(hCon
, c
, ' ', DEFAULT_ATTRIB
);
203 okCHAR(hCon
, c
, mytest
[5], TEST_ATTRIB
);
204 for (c
.X
= 1; c
.X
< 8; c
.X
++)
205 okCHAR(hCon
, c
, ' ', TEST_ATTRIB
);
206 okCHAR(hCon
, c
, mytest
[7], TEST_ATTRIB
);
208 okCHAR(hCon
, c
, ' ', DEFAULT_ATTRIB
);
212 /* write line, wrapping disabled, strings end on end of line */
213 c
.X
= sbSize
.X
- 4; c
.Y
= 0;
214 ok(SetConsoleCursorPosition(hCon
, c
) != 0, "Cursor in upper-left-4\n");
216 ok(WriteConsole(hCon
, mytest
, mylen
, &len
, NULL
) != 0 && len
== mylen
, "WriteConsole\n");
218 for (c
.X
= sbSize
.X
- 4; c
.X
< sbSize
.X
; c
.X
++)
220 okCHAR(hCon
, c
, mytest
[c
.X
- sbSize
.X
+ 4], TEST_ATTRIB
);
223 okCHAR(hCon
, c
, mytest
[5], TEST_ATTRIB
);
224 for (c
.X
= 1; c
.X
< 8; c
.X
++)
225 okCHAR(hCon
, c
, ' ', TEST_ATTRIB
);
226 okCHAR(hCon
, c
, mytest
[7], TEST_ATTRIB
);
228 okCHAR(hCon
, c
, ' ', DEFAULT_ATTRIB
);
232 /* write line, wrapping disabled, strings end after end of line */
233 c
.X
= sbSize
.X
- 3; c
.Y
= 0;
234 ok(SetConsoleCursorPosition(hCon
, c
) != 0, "Cursor in upper-left-4\n");
236 ok(WriteConsole(hCon
, mytest
, mylen
, &len
, NULL
) != 0 && len
== mylen
, "WriteConsole\n");
238 for (p
= mylen2
- 3; p
< mylen2
; p
++)
240 c
.X
= sbSize
.X
- 3 + p
% 3;
241 okCHAR(hCon
, c
, mytest
[p
], TEST_ATTRIB
);
244 okCHAR(hCon
, c
, mytest
[5], TEST_ATTRIB
);
245 for (c
.X
= 1; c
.X
< 8; c
.X
++)
246 okCHAR(hCon
, c
, ' ', TEST_ATTRIB
);
247 okCHAR(hCon
, c
, mytest
[7], TEST_ATTRIB
);
249 okCHAR(hCon
, c
, ' ', DEFAULT_ATTRIB
);
254 static void testWriteWrappedNotProcessed(HANDLE hCon
, COORD sbSize
)
258 const char* mytest
= "abcd\nf\tg";
259 const int mylen
= strlen(mytest
);
262 ok(GetConsoleMode(hCon
, &mode
) && SetConsoleMode(hCon
,(mode
| ENABLE_WRAP_AT_EOL_OUTPUT
) & ~(ENABLE_PROCESSED_OUTPUT
)),
263 "setting wrap at EOL & clearing processed output\n");
265 /* write line, wrapping enabled, buffer doesn't exceed sb width */
266 c
.X
= sbSize
.X
- 9; c
.Y
= 0;
267 ok(SetConsoleCursorPosition(hCon
, c
) != 0, "Cursor in upper-left-9\n");
269 ok(WriteConsole(hCon
, mytest
, mylen
, &len
, NULL
) != 0 && len
== mylen
, "WriteConsole\n");
271 for (p
= 0; p
< mylen
; p
++)
273 c
.X
= sbSize
.X
- 9 + p
;
274 okCHAR(hCon
, c
, mytest
[p
], TEST_ATTRIB
);
276 c
.X
= sbSize
.X
- 9 + mylen
;
277 okCHAR(hCon
, c
, ' ', DEFAULT_ATTRIB
);
279 okCHAR(hCon
, c
, ' ', DEFAULT_ATTRIB
);
281 /* write line, wrapping enabled, buffer does exceed sb width */
282 c
.X
= sbSize
.X
- 3; c
.Y
= 0;
283 ok(SetConsoleCursorPosition(hCon
, c
) != 0, "Cursor in upper-left-3\n");
287 okCHAR(hCon
, c
, ' ', DEFAULT_ATTRIB
);
290 static void testWriteWrappedProcessed(HANDLE hCon
, COORD sbSize
)
294 const char* mytest
= "abcd\nf\tg";
295 const int mylen
= strlen(mytest
);
298 ok(GetConsoleMode(hCon
, &mode
) && SetConsoleMode(hCon
, mode
| (ENABLE_WRAP_AT_EOL_OUTPUT
|ENABLE_PROCESSED_OUTPUT
)),
299 "setting wrap at EOL & processed output\n");
301 /* write line, wrapping enabled, buffer doesn't exceed sb width */
302 c
.X
= sbSize
.X
- 9; c
.Y
= 0;
303 ok(SetConsoleCursorPosition(hCon
, c
) != 0, "Cursor in upper-left-9\n");
305 ok(WriteConsole(hCon
, mytest
, mylen
, &len
, NULL
) != 0 && len
== mylen
, "WriteConsole\n");
306 for (p
= 0; p
< 4; p
++)
308 c
.X
= sbSize
.X
- 9 + p
;
309 okCHAR(hCon
, c
, mytest
[p
], TEST_ATTRIB
);
311 c
.X
= sbSize
.X
- 9 + p
;
312 okCHAR(hCon
, c
, ' ', DEFAULT_ATTRIB
);
314 okCHAR(hCon
, c
, mytest
[5], TEST_ATTRIB
);
315 for (c
.X
= 1; c
.X
< 8; c
.X
++)
316 okCHAR(hCon
, c
, ' ', TEST_ATTRIB
);
317 okCHAR(hCon
, c
, mytest
[7], TEST_ATTRIB
);
319 okCHAR(hCon
, c
, ' ', DEFAULT_ATTRIB
);
322 /* write line, wrapping enabled, buffer does exceed sb width */
323 c
.X
= sbSize
.X
- 3; c
.Y
= 2;
324 ok(SetConsoleCursorPosition(hCon
, c
) != 0, "Cursor in upper-left-3\n");
326 ok(WriteConsole(hCon
, mytest
, mylen
, &len
, NULL
) != 0 && len
== mylen
, "WriteConsole\n");
327 for (p
= 0; p
< 3; p
++)
329 c
.X
= sbSize
.X
- 3 + p
;
330 okCHAR(hCon
, c
, mytest
[p
], TEST_ATTRIB
);
333 okCHAR(hCon
, c
, mytest
[3], TEST_ATTRIB
);
335 okCHAR(hCon
, c
, ' ', DEFAULT_ATTRIB
);
338 okCHAR(hCon
, c
, mytest
[5], TEST_ATTRIB
);
339 for (c
.X
= 1; c
.X
< 8; c
.X
++)
340 okCHAR(hCon
, c
, ' ', TEST_ATTRIB
);
341 okCHAR(hCon
, c
, mytest
[7], TEST_ATTRIB
);
343 okCHAR(hCon
, c
, ' ', DEFAULT_ATTRIB
);
347 static void testWrite(HANDLE hCon
, COORD sbSize
)
349 /* FIXME: should in fact insure that the sb is at least 10 character wide */
350 ok(SetConsoleTextAttribute(hCon
, TEST_ATTRIB
), "Setting default text color\n");
351 resetContent(hCon
, sbSize
, FALSE
);
352 testWriteSimple(hCon
, sbSize
);
353 resetContent(hCon
, sbSize
, FALSE
);
354 testWriteNotWrappedNotProcessed(hCon
, sbSize
);
355 resetContent(hCon
, sbSize
, FALSE
);
356 testWriteNotWrappedProcessed(hCon
, sbSize
);
357 resetContent(hCon
, sbSize
, FALSE
);
358 testWriteWrappedNotProcessed(hCon
, sbSize
);
359 resetContent(hCon
, sbSize
, FALSE
);
360 testWriteWrappedProcessed(hCon
, sbSize
);
363 static void testScroll(HANDLE hCon
, COORD sbSize
)
365 SMALL_RECT scroll
, clip
;
372 /* no clipping, src & dst rect don't overlap */
373 resetContent(hCon
, sbSize
, TRUE
);
375 #define IN_SRECT(r,c) ((r).Left <= (c).X && (c).X <= (r).Right && (r).Top <= (c).Y && (c).Y <= (r).Bottom)
376 #define IN_SRECT2(r,d,c) ((d).X <= (c).X && (c).X <= (d).X + (r).Right - (r).Left && (d).Y <= (c).Y && (c).Y <= (d).Y + (r).Bottom - (r).Top)
379 scroll
.Right
= W
- 1;
381 scroll
.Bottom
= H
- 1;
384 ci
.Char
.UnicodeChar
= '#';
385 ci
.Attributes
= TEST_ATTRIB
;
388 clip
.Right
= sbSize
.X
- 1;
390 clip
.Bottom
= sbSize
.Y
- 1;
392 ok(ScrollConsoleScreenBuffer(hCon
, &scroll
, NULL
, dst
, &ci
), "Scrolling SB\n");
394 for (c
.Y
= 0; c
.Y
< sbSize
.Y
; c
.Y
++)
396 for (c
.X
= 0; c
.X
< sbSize
.X
; c
.X
++)
398 if (IN_SRECT2(scroll
, dst
, c
) && IN_SRECT(clip
, c
))
402 okCHAR(hCon
, c
, CONTENT(tc
), DEFAULT_ATTRIB
);
404 else if (IN_SRECT(scroll
, c
) && IN_SRECT(clip
, c
))
405 okCHAR(hCon
, c
, '#', TEST_ATTRIB
);
406 else okCHAR(hCon
, c
, CONTENT(c
), DEFAULT_ATTRIB
);
410 /* no clipping, src & dst rect do overlap */
411 resetContent(hCon
, sbSize
, TRUE
);
414 scroll
.Right
= W
- 1;
416 scroll
.Bottom
= H
- 1;
419 ci
.Char
.UnicodeChar
= '#';
420 ci
.Attributes
= TEST_ATTRIB
;
423 clip
.Right
= sbSize
.X
- 1;
425 clip
.Bottom
= sbSize
.Y
- 1;
427 ok(ScrollConsoleScreenBuffer(hCon
, &scroll
, NULL
, dst
, &ci
), "Scrolling SB\n");
429 for (c
.Y
= 0; c
.Y
< sbSize
.Y
; c
.Y
++)
431 for (c
.X
= 0; c
.X
< sbSize
.X
; c
.X
++)
433 if (dst
.X
<= c
.X
&& c
.X
< dst
.X
+ W
&& dst
.Y
<= c
.Y
&& c
.Y
< dst
.Y
+ H
)
437 okCHAR(hCon
, c
, CONTENT(tc
), DEFAULT_ATTRIB
);
439 else if (c
.X
< W
&& c
.Y
< H
) okCHAR(hCon
, c
, '#', TEST_ATTRIB
);
440 else okCHAR(hCon
, c
, CONTENT(c
), DEFAULT_ATTRIB
);
444 /* clipping, src & dst rect don't overlap */
445 resetContent(hCon
, sbSize
, TRUE
);
448 scroll
.Right
= W
- 1;
450 scroll
.Bottom
= H
- 1;
453 ci
.Char
.UnicodeChar
= '#';
454 ci
.Attributes
= TEST_ATTRIB
;
457 clip
.Right
= min(W
+ W
/ 2, sbSize
.X
- 1);
459 clip
.Bottom
= min(H
+ H
/ 2, sbSize
.Y
- 1);
461 ok(ScrollConsoleScreenBuffer(hCon
, &scroll
, &clip
, dst
, &ci
), "Scrolling SB\n");
463 for (c
.Y
= 0; c
.Y
< sbSize
.Y
; c
.Y
++)
465 for (c
.X
= 0; c
.X
< sbSize
.X
; c
.X
++)
467 if (IN_SRECT2(scroll
, dst
, c
) && IN_SRECT(clip
, c
))
471 okCHAR(hCon
, c
, CONTENT(tc
), DEFAULT_ATTRIB
);
473 else if (IN_SRECT(scroll
, c
) && IN_SRECT(clip
, c
))
474 okCHAR(hCon
, c
, '#', TEST_ATTRIB
);
475 else okCHAR(hCon
, c
, CONTENT(c
), DEFAULT_ATTRIB
);
480 /* clipping, src & dst rect do overlap */
481 resetContent(hCon
, sbSize
, TRUE
);
484 scroll
.Right
= W
- 1;
486 scroll
.Bottom
= H
- 1;
489 ci
.Char
.UnicodeChar
= '#';
490 ci
.Attributes
= TEST_ATTRIB
;
493 clip
.Right
= min(W
+ W
/ 2, sbSize
.X
- 1);
495 clip
.Bottom
= min(H
+ H
/ 2, sbSize
.Y
- 1);
497 ok(ScrollConsoleScreenBuffer(hCon
, &scroll
, &clip
, dst
, &ci
), "Scrolling SB\n");
499 for (c
.Y
= 0; c
.Y
< sbSize
.Y
; c
.Y
++)
501 for (c
.X
= 0; c
.X
< sbSize
.X
; c
.X
++)
503 if (IN_SRECT2(scroll
, dst
, c
) && IN_SRECT(clip
, c
))
507 okCHAR(hCon
, c
, CONTENT(tc
), DEFAULT_ATTRIB
);
509 else if (IN_SRECT(scroll
, c
) && IN_SRECT(clip
, c
))
510 okCHAR(hCon
, c
, '#', TEST_ATTRIB
);
511 else okCHAR(hCon
, c
, CONTENT(c
), DEFAULT_ATTRIB
);
517 static int mch_count
;
518 /* we need the event as Wine console event generation isn't synchronous
519 * (ie GenerateConsoleCtrlEvent returns before all ctrl-handlers in all
520 * processes have been called).
522 static HANDLE mch_event
;
523 static BOOL WINAPI
mch(DWORD event
)
530 static void testCtrlHandler(void)
532 ok(!SetConsoleCtrlHandler(mch
, FALSE
), "Shouldn't succeed\n");
533 ok(GetLastError() == ERROR_INVALID_PARAMETER
, "Bad error %lu\n", GetLastError());
534 ok(SetConsoleCtrlHandler(mch
, TRUE
), "Couldn't set handler\n");
535 /* wine requires the event for the test, as we cannot insure, so far, that event
536 * are processed synchronously in GenerateConsoleCtrlEvent()
538 mch_event
= CreateEventA(NULL
, TRUE
, FALSE
, NULL
);
540 ok(GenerateConsoleCtrlEvent(CTRL_C_EVENT
, 0), "Couldn't send ctrl-c event\n");
541 todo_wine
ok(mch_count
== 1, "Event isn't synchronous\n");
542 ok(WaitForSingleObject(mch_event
, 3000) == WAIT_OBJECT_0
, "event sending didn't work\n");
543 CloseHandle(mch_event
);
544 ok(SetConsoleCtrlHandler(NULL
, TRUE
), "Couldn't turn off ctrl-c handling\n");
545 mch_event
= CreateEventA(NULL
, TRUE
, FALSE
, NULL
);
547 ok(GenerateConsoleCtrlEvent(CTRL_C_EVENT
, 0), "Couldn't send ctrl-c event\n");
548 ok(WaitForSingleObject(mch_event
, 3000) == WAIT_TIMEOUT
&& mch_count
== 0, "Event shouldn't have been sent\n");
549 CloseHandle(mch_event
);
550 ok(SetConsoleCtrlHandler(mch
, FALSE
), "Couldn't remove handler\n");
551 ok(!SetConsoleCtrlHandler(mch
, FALSE
), "Shouldn't succeed\n");
552 ok(GetLastError() == ERROR_INVALID_PARAMETER
, "Bad error %lu\n", GetLastError());
557 HANDLE hConIn
, hConOut
;
559 CONSOLE_SCREEN_BUFFER_INFO sbi
;
561 /* be sure we have a clean console (and that's our own)
562 * FIXME: this will make the test fail (currently) if we don't run
564 * Another solution would be to rerun the test under wineconsole with
568 hConIn
= CreateFileA("CONIN$", GENERIC_READ
|GENERIC_WRITE
, 0, NULL
, OPEN_EXISTING
, 0, 0);
569 hConOut
= CreateFileA("CONOUT$", GENERIC_READ
|GENERIC_WRITE
, 0, NULL
, OPEN_EXISTING
, 0, 0);
571 /* first, we need to be sure we're attached to a console */
572 if (hConIn
== INVALID_HANDLE_VALUE
|| hConOut
== INVALID_HANDLE_VALUE
)
574 /* we're not attached to a console, let's do it */
576 hConIn
= CreateFileA("CONIN$", GENERIC_READ
|GENERIC_WRITE
, 0, NULL
, OPEN_EXISTING
, 0, 0);
577 hConOut
= CreateFileA("CONOUT$", GENERIC_READ
|GENERIC_WRITE
, 0, NULL
, OPEN_EXISTING
, 0, 0);
579 /* now verify everything's ok */
580 ok(hConIn
!= INVALID_HANDLE_VALUE
, "Opening ConIn\n");
581 ok(hConOut
!= INVALID_HANDLE_VALUE
, "Opening ConOut\n");
583 ok(ret
= GetConsoleScreenBufferInfo(hConOut
, &sbi
), "Getting sb info\n");
586 /* Non interactive tests */
587 testCursor(hConOut
, sbi
.dwSize
);
588 /* will test wrapped (on/off) & processed (on/off) strings output */
589 testWrite(hConOut
, sbi
.dwSize
);
590 /* will test line scrolling at the bottom of the screen */
591 /* testBottomScroll(); */
592 /* will test all the scrolling operations */
593 /* this one is disabled for now, Wine's result are way too bad */
594 testScroll(hConOut
, sbi
.dwSize
);
595 /* will test sb creation / modification... */
596 /* testScreenBuffer() */
598 /* still to be done: access rights & access on objects */