2 * Server-side console management
4 * Copyright (C) 1998 Alexandre Julliard
6 * Copyright 2020 Jacek Caban for CodeWeavers
9 * This library is free software; you can redistribute it and/or
10 * modify it under the terms of the GNU Lesser General Public
11 * License as published by the Free Software Foundation; either
12 * version 2.1 of the License, or (at your option) any later version.
14 * This library is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
17 * Lesser General Public License for more details.
19 * You should have received a copy of the GNU Lesser General Public
20 * License along with this library; if not, write to the Free Software
21 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
31 #include <sys/ioctl.h>
35 #define WIN32_NO_STATUS
43 #include "wine/condrv.h"
55 struct object obj
; /* object header */
56 int signaled
; /* is console signaled */
57 struct thread
*renderer
; /* console renderer thread */
58 struct screen_buffer
*active
; /* active screen buffer */
59 struct console_server
*server
; /* console server object */
60 unsigned int last_id
; /* id of last created console buffer */
61 struct fd
*fd
; /* for bare console, attached input fd */
62 struct async_queue ioctl_q
; /* ioctl queue */
63 struct async_queue read_q
; /* read queue */
66 static void console_dump( struct object
*obj
, int verbose
);
67 static void console_destroy( struct object
*obj
);
68 static int console_signaled( struct object
*obj
, struct wait_queue_entry
*entry
);
69 static struct fd
*console_get_fd( struct object
*obj
);
70 static struct object
*console_lookup_name( struct object
*obj
, struct unicode_str
*name
,
71 unsigned int attr
, struct object
*root
);
72 static struct object
*console_open_file( struct object
*obj
, unsigned int access
,
73 unsigned int sharing
, unsigned int options
);
74 static int console_add_queue( struct object
*obj
, struct wait_queue_entry
*entry
);
76 static const struct object_ops console_ops
=
78 sizeof(struct console
), /* size */
79 &file_type
, /* type */
80 console_dump
, /* dump */
81 console_add_queue
, /* add_queue */
82 remove_queue
, /* remove_queue */
83 console_signaled
, /* signaled */
84 no_satisfied
, /* satisfied */
85 no_signal
, /* signal */
86 console_get_fd
, /* get_fd */
87 default_map_access
, /* map_access */
88 default_get_sd
, /* get_sd */
89 default_set_sd
, /* set_sd */
90 no_get_full_name
, /* get_full_name */
91 console_lookup_name
, /* lookup_name */
92 no_link_name
, /* link_name */
93 NULL
, /* unlink_name */
94 console_open_file
, /* open_file */
95 no_kernel_obj_list
, /* get_kernel_obj_list */
96 no_close_handle
, /* close_handle */
97 console_destroy
/* destroy */
100 static enum server_fd_type
console_get_fd_type( struct fd
*fd
);
101 static void console_get_file_info( struct fd
*fd
, obj_handle_t handle
, unsigned int info_class
);
102 static void console_get_volume_info( struct fd
*fd
, struct async
*async
, unsigned int info_class
);
103 static void console_read( struct fd
*fd
, struct async
*async
, file_pos_t pos
);
104 static void console_flush( struct fd
*fd
, struct async
*async
);
105 static void console_ioctl( struct fd
*fd
, ioctl_code_t code
, struct async
*async
);
107 static const struct fd_ops console_fd_ops
=
109 default_fd_get_poll_events
, /* get_poll_events */
110 default_poll_event
, /* poll_event */
111 console_get_fd_type
, /* get_fd_type */
112 console_read
, /* read */
113 no_fd_write
, /* write */
114 console_flush
, /* flush */
115 console_get_file_info
, /* get_file_info */
116 console_get_volume_info
, /* get_volume_info */
117 console_ioctl
, /* ioctl */
118 default_fd_cancel_async
, /* cancel_async */
119 default_fd_queue_async
, /* queue_async */
120 default_fd_reselect_async
/* reselect_async */
123 struct console_host_ioctl
125 unsigned int code
; /* ioctl code */
126 int output
; /* output id for screen buffer ioctls */
127 struct async
*async
; /* ioctl async */
128 struct list entry
; /* list entry */
131 struct console_server
133 struct object obj
; /* object header */
134 struct fd
*fd
; /* pseudo-fd for ioctls */
135 struct console
*console
; /* attached console */
136 struct list queue
; /* ioctl queue */
137 struct list read_queue
; /* blocking read queue */
138 unsigned int busy
: 1; /* flag if server processing an ioctl */
139 unsigned int once_input
: 1; /* flag if input thread has already been requested */
140 int term_fd
; /* UNIX terminal fd */
141 struct termios termios
; /* original termios */
144 static void console_server_dump( struct object
*obj
, int verbose
);
145 static void console_server_destroy( struct object
*obj
);
146 static int console_server_signaled( struct object
*obj
, struct wait_queue_entry
*entry
);
147 static struct fd
*console_server_get_fd( struct object
*obj
);
148 static struct object
*console_server_lookup_name( struct object
*obj
, struct unicode_str
*name
,
149 unsigned int attr
, struct object
*root
);
150 static struct object
*console_server_open_file( struct object
*obj
, unsigned int access
,
151 unsigned int sharing
, unsigned int options
);
153 static const struct object_ops console_server_ops
=
155 sizeof(struct console_server
), /* size */
156 &file_type
, /* type */
157 console_server_dump
, /* dump */
158 add_queue
, /* add_queue */
159 remove_queue
, /* remove_queue */
160 console_server_signaled
, /* signaled */
161 no_satisfied
, /* satisfied */
162 no_signal
, /* signal */
163 console_server_get_fd
, /* get_fd */
164 default_map_access
, /* map_access */
165 default_get_sd
, /* get_sd */
166 default_set_sd
, /* set_sd */
167 no_get_full_name
, /* get_full_name */
168 console_server_lookup_name
, /* lookup_name */
169 no_link_name
, /* link_name */
170 NULL
, /* unlink_name */
171 console_server_open_file
, /* open_file */
172 no_kernel_obj_list
, /* get_kernel_obj_list */
173 no_close_handle
, /* close_handle */
174 console_server_destroy
/* destroy */
177 static void console_server_ioctl( struct fd
*fd
, ioctl_code_t code
, struct async
*async
);
179 static const struct fd_ops console_server_fd_ops
=
181 default_fd_get_poll_events
, /* get_poll_events */
182 default_poll_event
, /* poll_event */
183 console_get_fd_type
, /* get_fd_type */
184 no_fd_read
, /* read */
185 no_fd_write
, /* write */
186 no_fd_flush
, /* flush */
187 no_fd_get_file_info
, /* get_file_info */
188 no_fd_get_volume_info
, /* get_volume_info */
189 console_server_ioctl
, /* ioctl */
190 default_fd_cancel_async
, /* cancel_async */
191 default_fd_queue_async
, /* queue_async */
192 default_fd_reselect_async
/* reselect_async */
200 short int pitch_family
;
202 data_size_t face_len
;
207 struct object obj
; /* object header */
208 struct list entry
; /* entry in list of all screen buffers */
209 struct console
*input
; /* associated console input */
210 unsigned int id
; /* buffer id */
211 struct fd
*fd
; /* for bare console, attached output fd */
212 struct async_queue ioctl_q
; /* ioctl queue */
215 static void screen_buffer_dump( struct object
*obj
, int verbose
);
216 static void screen_buffer_destroy( struct object
*obj
);
217 static int screen_buffer_add_queue( struct object
*obj
, struct wait_queue_entry
*entry
);
218 static struct fd
*screen_buffer_get_fd( struct object
*obj
);
219 static struct object
*screen_buffer_open_file( struct object
*obj
, unsigned int access
,
220 unsigned int sharing
, unsigned int options
);
222 static const struct object_ops screen_buffer_ops
=
224 sizeof(struct screen_buffer
), /* size */
225 &file_type
, /* type */
226 screen_buffer_dump
, /* dump */
227 screen_buffer_add_queue
, /* add_queue */
228 NULL
, /* remove_queue */
230 NULL
, /* satisfied */
231 no_signal
, /* signal */
232 screen_buffer_get_fd
, /* get_fd */
233 default_map_access
, /* map_access */
234 default_get_sd
, /* get_sd */
235 default_set_sd
, /* set_sd */
236 no_get_full_name
, /* get_full_name */
237 no_lookup_name
, /* lookup_name */
238 no_link_name
, /* link_name */
239 NULL
, /* unlink_name */
240 screen_buffer_open_file
, /* open_file */
241 no_kernel_obj_list
, /* get_kernel_obj_list */
242 no_close_handle
, /* close_handle */
243 screen_buffer_destroy
/* destroy */
246 static void screen_buffer_write( struct fd
*fd
, struct async
*async
, file_pos_t pos
);
247 static void screen_buffer_ioctl( struct fd
*fd
, ioctl_code_t code
, struct async
*async
);
249 static const struct fd_ops screen_buffer_fd_ops
=
251 default_fd_get_poll_events
, /* get_poll_events */
252 default_poll_event
, /* poll_event */
253 console_get_fd_type
, /* get_fd_type */
254 no_fd_read
, /* read */
255 screen_buffer_write
, /* write */
256 no_fd_flush
, /* flush */
257 console_get_file_info
, /* get_file_info */
258 console_get_volume_info
, /* get_volume_info */
259 screen_buffer_ioctl
, /* ioctl */
260 default_fd_cancel_async
, /* cancel_async */
261 default_fd_queue_async
, /* queue_async */
262 default_fd_reselect_async
/* reselect_async */
265 static void console_device_dump( struct object
*obj
, int verbose
);
266 static struct object
*console_device_lookup_name( struct object
*obj
, struct unicode_str
*name
,
267 unsigned int attr
, struct object
*root
);
268 static struct object
*console_device_open_file( struct object
*obj
, unsigned int access
,
269 unsigned int sharing
, unsigned int options
);
271 static const struct object_ops console_device_ops
=
273 sizeof(struct object
), /* size */
274 &device_type
, /* type */
275 console_device_dump
, /* dump */
276 no_add_queue
, /* add_queue */
277 NULL
, /* remove_queue */
279 no_satisfied
, /* satisfied */
280 no_signal
, /* signal */
281 no_get_fd
, /* get_fd */
282 default_map_access
, /* map_access */
283 default_get_sd
, /* get_sd */
284 default_set_sd
, /* set_sd */
285 default_get_full_name
, /* get_full_name */
286 console_device_lookup_name
, /* lookup_name */
287 directory_link_name
, /* link_name */
288 default_unlink_name
, /* unlink_name */
289 console_device_open_file
, /* open_file */
290 no_kernel_obj_list
, /* get_kernel_obj_list */
291 no_close_handle
, /* close_handle */
292 no_destroy
/* destroy */
297 struct object obj
; /* object header */
298 struct fd
*fd
; /* pseudo-fd */
301 static void console_input_dump( struct object
*obj
, int verbose
);
302 static struct object
*console_input_open_file( struct object
*obj
, unsigned int access
,
303 unsigned int sharing
, unsigned int options
);
304 static int console_input_add_queue( struct object
*obj
, struct wait_queue_entry
*entry
);
305 static struct fd
*console_input_get_fd( struct object
*obj
);
306 static void console_input_destroy( struct object
*obj
);
308 static const struct object_ops console_input_ops
=
310 sizeof(struct console_input
), /* size */
311 &device_type
, /* type */
312 console_input_dump
, /* dump */
313 console_input_add_queue
, /* add_queue */
314 NULL
, /* remove_queue */
316 no_satisfied
, /* satisfied */
317 no_signal
, /* signal */
318 console_input_get_fd
, /* get_fd */
319 default_map_access
, /* map_access */
320 default_get_sd
, /* get_sd */
321 default_set_sd
, /* set_sd */
322 no_get_full_name
, /* get_full_name */
323 no_lookup_name
, /* lookup_name */
324 directory_link_name
, /* link_name */
325 default_unlink_name
, /* unlink_name */
326 console_input_open_file
, /* open_file */
327 no_kernel_obj_list
, /* get_kernel_obj_list */
328 no_close_handle
, /* close_handle */
329 console_input_destroy
/* destroy */
332 static void console_input_read( struct fd
*fd
, struct async
*async
, file_pos_t pos
);
333 static void console_input_flush( struct fd
*fd
, struct async
*async
);
334 static void console_input_ioctl( struct fd
*fd
, ioctl_code_t code
, struct async
*async
);
336 static const struct fd_ops console_input_fd_ops
=
338 default_fd_get_poll_events
, /* get_poll_events */
339 default_poll_event
, /* poll_event */
340 console_get_fd_type
, /* get_fd_type */
341 console_input_read
, /* read */
342 no_fd_write
, /* write */
343 console_input_flush
, /* flush */
344 console_get_file_info
, /* get_file_info */
345 console_get_volume_info
, /* get_volume_info */
346 console_input_ioctl
, /* ioctl */
347 default_fd_cancel_async
, /* cancel_async */
348 default_fd_queue_async
, /* queue_async */
349 default_fd_reselect_async
/* reselect_async */
352 struct console_output
354 struct object obj
; /* object header */
355 struct fd
*fd
; /* pseudo-fd */
358 static void console_output_dump( struct object
*obj
, int verbose
);
359 static int console_output_add_queue( struct object
*obj
, struct wait_queue_entry
*entry
);
360 static struct fd
*console_output_get_fd( struct object
*obj
);
361 static struct object
*console_output_open_file( struct object
*obj
, unsigned int access
,
362 unsigned int sharing
, unsigned int options
);
363 static void console_output_destroy( struct object
*obj
);
365 static const struct object_ops console_output_ops
=
367 sizeof(struct console_output
), /* size */
368 &device_type
, /* type */
369 console_output_dump
, /* dump */
370 console_output_add_queue
, /* add_queue */
371 NULL
, /* remove_queue */
373 no_satisfied
, /* satisfied */
374 no_signal
, /* signal */
375 console_output_get_fd
, /* get_fd */
376 default_map_access
, /* map_access */
377 default_get_sd
, /* get_sd */
378 default_set_sd
, /* set_sd */
379 no_get_full_name
, /* get_full_name */
380 no_lookup_name
, /* lookup_name */
381 directory_link_name
, /* link_name */
382 default_unlink_name
, /* unlink_name */
383 console_output_open_file
, /* open_file */
384 no_kernel_obj_list
, /* get_kernel_obj_list */
385 no_close_handle
, /* close_handle */
386 console_output_destroy
/* destroy */
389 static void console_output_write( struct fd
*fd
, struct async
*async
, file_pos_t pos
);
390 static void console_output_ioctl( struct fd
*fd
, ioctl_code_t code
, struct async
*async
);
392 static const struct fd_ops console_output_fd_ops
=
394 default_fd_get_poll_events
, /* get_poll_events */
395 default_poll_event
, /* poll_event */
396 console_get_fd_type
, /* get_fd_type */
397 no_fd_read
, /* read */
398 console_output_write
, /* write */
399 no_fd_flush
, /* flush */
400 console_get_file_info
, /* get_file_info */
401 console_get_volume_info
, /* get_volume_info */
402 console_output_ioctl
, /* ioctl */
403 default_fd_cancel_async
, /* cancel_async */
404 default_fd_queue_async
, /* queue_async */
405 default_fd_reselect_async
/* reselect_async */
408 struct console_connection
410 struct object obj
; /* object header */
411 struct fd
*fd
; /* pseudo-fd for ioctls */
414 static void console_connection_dump( struct object
*obj
, int verbose
);
415 static struct fd
*console_connection_get_fd( struct object
*obj
);
416 static struct object
*console_connection_lookup_name( struct object
*obj
, struct unicode_str
*name
,
417 unsigned int attr
, struct object
*root
);
418 static struct object
*console_connection_open_file( struct object
*obj
, unsigned int access
,
419 unsigned int sharing
, unsigned int options
);
420 static int console_connection_close_handle( struct object
*obj
, struct process
*process
, obj_handle_t handle
);
421 static void console_connection_destroy( struct object
*obj
);
423 static const struct object_ops console_connection_ops
=
425 sizeof(struct console_connection
),/* size */
426 &device_type
, /* type */
427 console_connection_dump
, /* dump */
428 no_add_queue
, /* add_queue */
429 NULL
, /* remove_queue */
431 no_satisfied
, /* satisfied */
432 no_signal
, /* signal */
433 console_connection_get_fd
, /* get_fd */
434 default_map_access
, /* map_access */
435 default_get_sd
, /* get_sd */
436 default_set_sd
, /* set_sd */
437 no_get_full_name
, /* get_full_name */
438 console_connection_lookup_name
, /* lookup_name */
439 directory_link_name
, /* link_name */
440 default_unlink_name
, /* unlink_name */
441 console_connection_open_file
, /* open_file */
442 no_kernel_obj_list
, /* get_kernel_obj_list */
443 console_connection_close_handle
, /* close_handle */
444 console_connection_destroy
/* destroy */
447 static void console_connection_ioctl( struct fd
*fd
, ioctl_code_t code
, struct async
*async
);
449 static const struct fd_ops console_connection_fd_ops
=
451 default_fd_get_poll_events
, /* get_poll_events */
452 default_poll_event
, /* poll_event */
453 console_get_fd_type
, /* get_fd_type */
454 no_fd_read
, /* read */
455 no_fd_write
, /* write */
456 no_fd_flush
, /* flush */
457 no_fd_get_file_info
, /* get_file_info */
458 no_fd_get_volume_info
, /* get_volume_info */
459 console_connection_ioctl
, /* ioctl */
460 default_fd_cancel_async
, /* cancel_async */
461 default_fd_queue_async
, /* queue_async */
462 default_fd_reselect_async
/* reselect_async */
465 static struct list screen_buffer_list
= LIST_INIT(screen_buffer_list
);
467 static int queue_host_ioctl( struct console_server
*server
, unsigned int code
, unsigned int output
,
468 struct async
*async
, struct async_queue
*queue
);
470 static int console_add_queue( struct object
*obj
, struct wait_queue_entry
*entry
)
472 struct console
*console
= (struct console
*)obj
;
473 assert( obj
->ops
== &console_ops
);
474 /* before waiting, ensure conhost's input thread has been started */
475 if (console
->server
&& !console
->server
->once_input
)
477 console
->server
->once_input
= 1;
478 if (console
->server
->term_fd
== -1)
479 queue_host_ioctl( console
->server
, IOCTL_CONDRV_PEEK
, 0, NULL
, NULL
);
481 return add_queue( &console
->obj
, entry
);
484 static int console_signaled( struct object
*obj
, struct wait_queue_entry
*entry
)
486 struct console
*console
= (struct console
*)obj
;
487 return console
->signaled
;
490 static struct fd
*console_get_fd( struct object
*obj
)
492 struct console
*console
= (struct console
*)obj
;
493 assert( obj
->ops
== &console_ops
);
494 return (struct fd
*)grab_object( console
->fd
);
497 static enum server_fd_type
console_get_fd_type( struct fd
*fd
)
502 static void console_get_file_info( struct fd
*fd
, obj_handle_t handle
, unsigned int info_class
)
504 set_error( STATUS_INVALID_DEVICE_REQUEST
);
507 static void console_get_volume_info( struct fd
*fd
, struct async
*async
, unsigned int info_class
)
511 case FileFsDeviceInformation
:
513 static const FILE_FS_DEVICE_INFORMATION device_info
=
516 FILE_DEVICE_ALLOW_APPCONTAINER_TRAVERSAL
518 if (get_reply_max_size() >= sizeof(device_info
))
519 set_reply_data( &device_info
, sizeof(device_info
) );
521 set_error( STATUS_BUFFER_TOO_SMALL
);
525 set_error( STATUS_NOT_IMPLEMENTED
);
529 static struct object
*create_console(void)
531 struct console
*console
;
533 if (!(console
= alloc_object( &console_ops
)))
536 console
->renderer
= NULL
;
537 console
->signaled
= 0;
538 console
->active
= NULL
;
539 console
->server
= NULL
;
541 console
->last_id
= 0;
542 init_async_queue( &console
->ioctl_q
);
543 init_async_queue( &console
->read_q
);
545 console
->fd
= alloc_pseudo_fd( &console_fd_ops
, &console
->obj
, FILE_SYNCHRONOUS_IO_NONALERT
);
548 release_object( console
);
551 allow_fd_caching( console
->fd
);
552 return &console
->obj
;
555 static void console_host_ioctl_terminate( struct console_host_ioctl
*call
, unsigned int status
)
559 async_terminate( call
->async
, status
);
560 release_object( call
->async
);
565 static int queue_host_ioctl( struct console_server
*server
, unsigned int code
, unsigned int output
,
566 struct async
*async
, struct async_queue
*queue
)
568 struct console_host_ioctl
*ioctl
;
570 if (!(ioctl
= mem_alloc( sizeof(*ioctl
) ))) return 0;
572 ioctl
->output
= output
;
576 ioctl
->async
= (struct async
*)grab_object( async
);
577 queue_async( queue
, async
);
579 list_add_tail( &server
->queue
, &ioctl
->entry
);
580 wake_up( &server
->obj
, 0 );
581 if (async
) set_error( STATUS_PENDING
);
585 static void disconnect_console_server( struct console_server
*server
)
587 while (!list_empty( &server
->queue
))
589 struct console_host_ioctl
*call
= LIST_ENTRY( list_head( &server
->queue
), struct console_host_ioctl
, entry
);
590 list_remove( &call
->entry
);
591 console_host_ioctl_terminate( call
, STATUS_CANCELLED
);
593 while (!list_empty( &server
->read_queue
))
595 struct console_host_ioctl
*call
= LIST_ENTRY( list_head( &server
->read_queue
), struct console_host_ioctl
, entry
);
596 list_remove( &call
->entry
);
597 console_host_ioctl_terminate( call
, STATUS_CANCELLED
);
600 if (server
->term_fd
!= -1)
602 tcsetattr( server
->term_fd
, TCSANOW
, &server
->termios
);
603 close( server
->term_fd
);
604 server
->term_fd
= -1;
609 assert( server
->console
->server
== server
);
610 server
->console
->server
= NULL
;
611 server
->console
= NULL
;
612 wake_up( &server
->obj
, 0 );
616 static void set_active_screen_buffer( struct console
*console
, struct screen_buffer
*screen_buffer
)
618 if (console
->active
== screen_buffer
) return;
619 if (console
->active
) release_object( console
->active
);
620 console
->active
= (struct screen_buffer
*)grab_object( screen_buffer
);
622 if (console
->server
) queue_host_ioctl( console
->server
, IOCTL_CONDRV_ACTIVATE
,
623 screen_buffer
->id
, NULL
, NULL
);
626 static struct object
*create_screen_buffer( struct console
*console
)
628 struct screen_buffer
*screen_buffer
;
630 if (console
->last_id
== ~0)
632 set_error( STATUS_NO_MEMORY
);
636 if (!(screen_buffer
= alloc_object( &screen_buffer_ops
)))
639 screen_buffer
->id
= ++console
->last_id
;
640 screen_buffer
->input
= console
;
641 init_async_queue( &screen_buffer
->ioctl_q
);
642 list_add_head( &screen_buffer_list
, &screen_buffer
->entry
);
644 screen_buffer
->fd
= alloc_pseudo_fd( &screen_buffer_fd_ops
, &screen_buffer
->obj
,
645 FILE_SYNCHRONOUS_IO_NONALERT
);
646 if (!screen_buffer
->fd
)
648 release_object( screen_buffer
);
651 allow_fd_caching(screen_buffer
->fd
);
653 if (console
->server
) queue_host_ioctl( console
->server
, IOCTL_CONDRV_INIT_OUTPUT
,
654 screen_buffer
->id
, NULL
, NULL
);
655 if (!console
->active
) set_active_screen_buffer( console
, screen_buffer
);
656 return &screen_buffer
->obj
;
659 struct thread
*console_get_renderer( struct console
*console
)
661 return console
->renderer
;
664 struct console_signal_info
666 struct console
*console
;
671 static int propagate_console_signal_cb(struct process
*process
, void *user
)
673 struct console_signal_info
* csi
= (struct console_signal_info
*)user
;
675 if (process
->console
== csi
->console
&& (!csi
->group
|| process
->group_id
== csi
->group
))
677 /* find a suitable thread to signal */
678 struct thread
*thread
;
679 LIST_FOR_EACH_ENTRY( thread
, &process
->thread_list
, struct thread
, proc_entry
)
681 if (send_thread_signal( thread
, csi
->signal
)) break;
687 static void propagate_console_signal( struct console
*console
,
688 int sig
, process_id_t group_id
)
690 struct console_signal_info csi
;
694 set_error( STATUS_INVALID_PARAMETER
);
699 case CTRL_C_EVENT
: csi
.signal
= SIGINT
; break;
700 case CTRL_BREAK_EVENT
: csi
.signal
= SIGQUIT
; break;
702 /* FIXME: should support the other events */
703 set_error( STATUS_NOT_IMPLEMENTED
);
706 csi
.console
= console
;
707 csi
.group
= group_id
;
709 enum_processes(propagate_console_signal_cb
, &csi
);
712 struct console_process_list
716 process_id_t
*processes
;
717 struct console
*console
;
720 static int console_process_list_cb(struct process
*process
, void *user
)
722 struct console_process_list
*cpl
= user
;
724 if (process
->console
== cpl
->console
)
726 if (cpl
->count
< cpl
->size
) cpl
->processes
[cpl
->count
] = process
->id
;
734 static void console_dump( struct object
*obj
, int verbose
)
736 struct console
*console
= (struct console
*)obj
;
737 assert( obj
->ops
== &console_ops
);
738 fprintf( stderr
, "Console input active=%p server=%p\n",
739 console
->active
, console
->server
);
742 static void console_destroy( struct object
*obj
)
744 struct console
*console
= (struct console
*)obj
;
745 struct screen_buffer
*curr
;
747 assert( obj
->ops
== &console_ops
);
751 assert( console
->server
->console
== console
);
752 disconnect_console_server( console
->server
);
755 if (console
->active
) release_object( console
->active
);
756 console
->active
= NULL
;
758 LIST_FOR_EACH_ENTRY( curr
, &screen_buffer_list
, struct screen_buffer
, entry
)
760 if (curr
->input
== console
) curr
->input
= NULL
;
763 free_async_queue( &console
->ioctl_q
);
764 free_async_queue( &console
->read_q
);
766 release_object( console
->fd
);
769 static struct object
*create_console_connection( struct console
*console
)
771 struct console_connection
*connection
;
773 if (current
->process
->console
)
775 set_error( STATUS_ACCESS_DENIED
);
779 if (!(connection
= alloc_object( &console_connection_ops
))) return NULL
;
780 if (!(connection
->fd
= alloc_pseudo_fd( &console_connection_fd_ops
, &connection
->obj
, 0 )))
782 release_object( connection
);
787 current
->process
->console
= (struct console
*)grab_object( console
);
789 return &connection
->obj
;
792 static struct object
*console_lookup_name( struct object
*obj
, struct unicode_str
*name
,
793 unsigned int attr
, struct object
*root
)
795 struct console
*console
= (struct console
*)obj
;
796 static const WCHAR connectionW
[] = {'C','o','n','n','e','c','t','i','o','n'};
797 assert( obj
->ops
== &console_ops
);
799 if (name
->len
== sizeof(connectionW
) && !memcmp( name
->str
, connectionW
, name
->len
))
802 return create_console_connection( console
);
808 static struct object
*console_open_file( struct object
*obj
, unsigned int access
,
809 unsigned int sharing
, unsigned int options
)
811 return grab_object( obj
);
814 static void screen_buffer_dump( struct object
*obj
, int verbose
)
816 struct screen_buffer
*screen_buffer
= (struct screen_buffer
*)obj
;
817 assert( obj
->ops
== &screen_buffer_ops
);
819 fprintf(stderr
, "Console screen buffer input=%p\n", screen_buffer
->input
);
822 static void screen_buffer_destroy( struct object
*obj
)
824 struct screen_buffer
*screen_buffer
= (struct screen_buffer
*)obj
;
826 assert( obj
->ops
== &screen_buffer_ops
);
828 list_remove( &screen_buffer
->entry
);
829 if (screen_buffer
->input
&& screen_buffer
->input
->server
)
830 queue_host_ioctl( screen_buffer
->input
->server
, IOCTL_CONDRV_CLOSE_OUTPUT
,
831 screen_buffer
->id
, NULL
, NULL
);
832 if (screen_buffer
->fd
) release_object( screen_buffer
->fd
);
833 free_async_queue( &screen_buffer
->ioctl_q
);
836 static struct object
*screen_buffer_open_file( struct object
*obj
, unsigned int access
,
837 unsigned int sharing
, unsigned int options
)
839 return grab_object( obj
);
842 static int screen_buffer_add_queue( struct object
*obj
, struct wait_queue_entry
*entry
)
844 struct screen_buffer
*screen_buffer
= (struct screen_buffer
*)obj
;
845 if (!screen_buffer
->input
)
847 set_error( STATUS_ACCESS_DENIED
);
850 return console_add_queue( &screen_buffer
->input
->obj
, entry
);
853 static struct fd
*screen_buffer_get_fd( struct object
*obj
)
855 struct screen_buffer
*screen_buffer
= (struct screen_buffer
*)obj
;
856 assert( obj
->ops
== &screen_buffer_ops
);
857 if (screen_buffer
->fd
)
858 return (struct fd
*)grab_object( screen_buffer
->fd
);
859 set_error( STATUS_OBJECT_TYPE_MISMATCH
);
863 static void console_server_dump( struct object
*obj
, int verbose
)
865 assert( obj
->ops
== &console_server_ops
);
866 fprintf( stderr
, "Console server\n" );
869 static void console_server_destroy( struct object
*obj
)
871 struct console_server
*server
= (struct console_server
*)obj
;
872 assert( obj
->ops
== &console_server_ops
);
873 disconnect_console_server( server
);
874 if (server
->fd
) release_object( server
->fd
);
877 static struct object
*console_server_lookup_name( struct object
*obj
, struct unicode_str
*name
,
878 unsigned int attr
, struct object
*root
)
880 struct console_server
*server
= (struct console_server
*)obj
;
881 static const WCHAR referenceW
[] = {'R','e','f','e','r','e','n','c','e'};
882 assert( obj
->ops
== &console_server_ops
);
884 if (name
->len
== sizeof(referenceW
) && !memcmp( name
->str
, referenceW
, name
->len
))
886 struct screen_buffer
*screen_buffer
;
890 set_error( STATUS_INVALID_HANDLE
);
893 if (!(server
->console
= (struct console
*)create_console())) return NULL
;
894 if (!(screen_buffer
= (struct screen_buffer
*)create_screen_buffer( server
->console
)))
896 release_object( server
->console
);
897 server
->console
= NULL
;
900 release_object( screen_buffer
);
901 server
->console
->server
= server
;
903 return &server
->console
->obj
;
909 static int console_server_signaled( struct object
*obj
, struct wait_queue_entry
*entry
)
911 struct console_server
*server
= (struct console_server
*)obj
;
912 assert( obj
->ops
== &console_server_ops
);
913 return !server
->console
|| !list_empty( &server
->queue
);
916 static struct fd
*console_server_get_fd( struct object
* obj
)
918 struct console_server
*server
= (struct console_server
*)obj
;
919 assert( obj
->ops
== &console_server_ops
);
920 return (struct fd
*)grab_object( server
->fd
);
923 static struct object
*console_server_open_file( struct object
*obj
, unsigned int access
,
924 unsigned int sharing
, unsigned int options
)
926 return grab_object( obj
);
929 static struct object
*create_console_server( void )
931 struct console_server
*server
;
933 if (!(server
= alloc_object( &console_server_ops
))) return NULL
;
934 server
->console
= NULL
;
936 server
->once_input
= 0;
937 server
->term_fd
= -1;
938 list_init( &server
->queue
);
939 list_init( &server
->read_queue
);
940 server
->fd
= alloc_pseudo_fd( &console_server_fd_ops
, &server
->obj
, FILE_SYNCHRONOUS_IO_NONALERT
);
943 release_object( server
);
946 allow_fd_caching(server
->fd
);
951 static int is_blocking_read_ioctl( unsigned int code
)
955 case IOCTL_CONDRV_READ_INPUT
:
956 case IOCTL_CONDRV_READ_CONSOLE
:
957 case IOCTL_CONDRV_READ_CONSOLE_CONTROL
:
958 case IOCTL_CONDRV_READ_FILE
:
965 static void console_ioctl( struct fd
*fd
, ioctl_code_t code
, struct async
*async
)
967 struct console
*console
= get_fd_user( fd
);
971 case IOCTL_CONDRV_CTRL_EVENT
:
973 const struct condrv_ctrl_event
*event
= get_req_data();
975 if (get_req_data_size() != sizeof(*event
))
977 set_error( STATUS_INVALID_PARAMETER
);
980 group
= event
->group_id
? event
->group_id
: current
->process
->group_id
;
983 set_error( STATUS_INVALID_PARAMETER
);
986 propagate_console_signal( console
, event
->event
, group
);
990 case IOCTL_CONDRV_GET_PROCESS_LIST
:
992 struct console_process_list cpl
;
993 if (get_reply_max_size() < sizeof(unsigned int))
995 set_error( STATUS_INVALID_PARAMETER
);
1001 cpl
.console
= console
;
1002 enum_processes( console_process_list_cb
, &cpl
);
1003 if (cpl
.count
* sizeof(process_id_t
) > get_reply_max_size())
1005 set_reply_data( &cpl
.count
, sizeof(cpl
.count
) );
1006 set_error( STATUS_BUFFER_TOO_SMALL
);
1010 cpl
.size
= cpl
.count
;
1012 if ((cpl
.processes
= set_reply_data_size( cpl
.size
* sizeof(process_id_t
) )))
1013 enum_processes( console_process_list_cb
, &cpl
);
1018 if (!console
->server
|| code
>> 16 != FILE_DEVICE_CONSOLE
)
1020 set_error( STATUS_INVALID_HANDLE
);
1023 queue_host_ioctl( console
->server
, code
, 0, async
, &console
->ioctl_q
);
1027 static void console_read( struct fd
*fd
, struct async
*async
, file_pos_t pos
)
1029 struct console
*console
= get_fd_user( fd
);
1031 if (!console
->server
)
1033 set_error( STATUS_INVALID_HANDLE
);
1036 queue_host_ioctl( console
->server
, IOCTL_CONDRV_READ_FILE
, 0, async
, &console
->ioctl_q
);
1039 static void console_flush( struct fd
*fd
, struct async
*async
)
1041 struct console
*console
= get_fd_user( fd
);
1043 if (!console
->server
)
1045 set_error( STATUS_INVALID_HANDLE
);
1048 queue_host_ioctl( console
->server
, IOCTL_CONDRV_FLUSH
, 0, NULL
, NULL
);
1051 static void screen_buffer_write( struct fd
*fd
, struct async
*async
, file_pos_t pos
)
1053 struct screen_buffer
*screen_buffer
= get_fd_user( fd
);
1055 if (!screen_buffer
->input
|| !screen_buffer
->input
->server
)
1057 set_error( STATUS_INVALID_HANDLE
);
1061 queue_host_ioctl( screen_buffer
->input
->server
, IOCTL_CONDRV_WRITE_FILE
,
1062 screen_buffer
->id
, async
, &screen_buffer
->ioctl_q
);
1065 static void screen_buffer_ioctl( struct fd
*fd
, ioctl_code_t code
, struct async
*async
)
1067 struct screen_buffer
*screen_buffer
= get_fd_user( fd
);
1071 case IOCTL_CONDRV_ACTIVATE
:
1072 if (!screen_buffer
->input
)
1074 set_error( STATUS_INVALID_HANDLE
);
1078 set_active_screen_buffer( screen_buffer
->input
, screen_buffer
);
1082 if (!screen_buffer
->input
|| !screen_buffer
->input
->server
|| code
>> 16 != FILE_DEVICE_CONSOLE
||
1083 is_blocking_read_ioctl( code
))
1085 set_error( STATUS_INVALID_HANDLE
);
1088 queue_host_ioctl( screen_buffer
->input
->server
, code
, screen_buffer
->id
,
1089 async
, &screen_buffer
->ioctl_q
);
1093 static void console_connection_ioctl( struct fd
*fd
, ioctl_code_t code
, struct async
*async
)
1095 struct console_connection
*console_connection
= get_fd_user( fd
);
1099 case IOCTL_CONDRV_BIND_PID
:
1101 struct process
*process
;
1103 if (get_req_data_size() != sizeof(unsigned int))
1105 set_error( STATUS_INVALID_PARAMETER
);
1108 if (current
->process
->console
)
1110 set_error( STATUS_INVALID_HANDLE
);
1114 pid
= *(unsigned int *)get_req_data();
1115 if (pid
== ATTACH_PARENT_PROCESS
) pid
= current
->process
->parent_id
;
1116 if (!(process
= get_process_from_id( pid
))) return;
1118 if (process
->console
)
1119 current
->process
->console
= (struct console
*)grab_object( process
->console
);
1120 else set_error( STATUS_ACCESS_DENIED
);
1121 release_object( process
);
1126 default_fd_ioctl( console_connection
->fd
, code
, async
);
1130 static void console_server_ioctl( struct fd
*fd
, ioctl_code_t code
, struct async
*async
)
1132 struct console_server
*server
= get_fd_user( fd
);
1136 case IOCTL_CONDRV_CTRL_EVENT
:
1138 const struct condrv_ctrl_event
*event
= get_req_data();
1139 if (get_req_data_size() != sizeof(*event
))
1141 set_error( STATUS_INVALID_PARAMETER
);
1144 if (!server
->console
)
1146 set_error( STATUS_INVALID_HANDLE
);
1149 propagate_console_signal( server
->console
, event
->event
, event
->group_id
);
1153 case IOCTL_CONDRV_SETUP_INPUT
:
1155 struct termios term
;
1156 obj_handle_t handle
;
1160 if (get_req_data_size() != sizeof(unsigned int) || get_reply_max_size())
1162 set_error( STATUS_INVALID_PARAMETER
);
1165 if (server
->term_fd
!= -1)
1167 tcsetattr( server
->term_fd
, TCSANOW
, &server
->termios
);
1168 close( server
->term_fd
);
1169 server
->term_fd
= -1;
1171 handle
= *(unsigned int *)get_req_data();
1172 if (!handle
) return;
1173 if (!(file
= get_file_obj( current
->process
, handle
, FILE_READ_DATA
)))
1177 unix_fd
= get_file_unix_fd( file
);
1178 release_object( file
);
1180 if (tcgetattr( unix_fd
, &server
->termios
))
1185 term
= server
->termios
;
1186 term
.c_lflag
&= ~(ECHO
| ECHONL
| ICANON
| IEXTEN
| ISIG
);
1187 term
.c_iflag
&= ~(BRKINT
| ICRNL
| INPCK
| ISTRIP
| IXON
);
1188 term
.c_cflag
&= ~(CSIZE
| PARENB
);
1189 term
.c_cflag
|= CS8
;
1190 term
.c_cc
[VMIN
] = 1;
1191 term
.c_cc
[VTIME
] = 0;
1192 if (tcsetattr( unix_fd
, TCSANOW
, &term
) || (server
->term_fd
= dup( unix_fd
)) == -1)
1198 set_error( STATUS_INVALID_HANDLE
);
1203 static void console_connection_dump( struct object
*obj
, int verbose
)
1205 fputs( "console connection\n", stderr
);
1208 static struct fd
*console_connection_get_fd( struct object
*obj
)
1210 struct console_connection
*connection
= (struct console_connection
*)obj
;
1211 return (struct fd
*)grab_object( connection
->fd
);
1214 static struct object
*console_connection_lookup_name( struct object
*obj
, struct unicode_str
*name
,
1215 unsigned int attr
, struct object
*root
)
1217 static const WCHAR referenceW
[] = {'R','e','f','e','r','e','n','c','e'};
1219 if (name
->len
== sizeof(referenceW
) && !memcmp( name
->str
, referenceW
, name
->len
))
1221 if (!current
->process
->console
)
1223 set_error( STATUS_INVALID_HANDLE
);
1227 return grab_object( current
->process
->console
);
1233 static struct object
*console_connection_open_file( struct object
*obj
, unsigned int access
,
1234 unsigned int sharing
, unsigned int options
)
1236 return grab_object( obj
);
1239 static int console_connection_close_handle( struct object
*obj
, struct process
*process
, obj_handle_t handle
)
1241 struct console
*console
= process
->console
;
1245 process
->console
= NULL
;
1246 release_object( console
);
1251 static void console_connection_destroy( struct object
*obj
)
1253 struct console_connection
*connection
= (struct console_connection
*)obj
;
1254 if (connection
->fd
) release_object( connection
->fd
);
1257 static void console_device_dump( struct object
*obj
, int verbose
)
1259 fputs( "Console device\n", stderr
);
1262 static struct object
*console_device_lookup_name( struct object
*obj
, struct unicode_str
*name
,
1263 unsigned int attr
, struct object
*root
)
1265 static const WCHAR connectionW
[] = {'C','o','n','n','e','c','t','i','o','n'};
1266 static const WCHAR consoleW
[] = {'C','o','n','s','o','l','e'};
1267 static const WCHAR current_inW
[] = {'C','u','r','r','e','n','t','I','n'};
1268 static const WCHAR current_outW
[] = {'C','u','r','r','e','n','t','O','u','t'};
1269 static const WCHAR inputW
[] = {'I','n','p','u','t'};
1270 static const WCHAR outputW
[] = {'O','u','t','p','u','t'};
1271 static const WCHAR screen_bufferW
[] = {'S','c','r','e','e','n','B','u','f','f','e','r'};
1272 static const WCHAR serverW
[] = {'S','e','r','v','e','r'};
1274 if (name
->len
== sizeof(current_inW
) && !memcmp( name
->str
, current_inW
, name
->len
))
1276 if (!current
->process
->console
)
1278 set_error( STATUS_INVALID_HANDLE
);
1282 return grab_object( current
->process
->console
);
1285 if (name
->len
== sizeof(current_outW
) && !memcmp( name
->str
, current_outW
, name
->len
))
1287 if (!current
->process
->console
|| !current
->process
->console
->active
)
1289 set_error( STATUS_INVALID_HANDLE
);
1293 return grab_object( current
->process
->console
->active
);
1296 if (name
->len
== sizeof(consoleW
) && !memcmp( name
->str
, consoleW
, name
->len
))
1299 return grab_object( obj
);
1302 if (name
->len
== sizeof(inputW
) && !memcmp( name
->str
, inputW
, name
->len
))
1304 struct console_input
*console_input
;
1306 if (!(console_input
= alloc_object( &console_input_ops
))) return NULL
;
1307 console_input
->fd
= alloc_pseudo_fd( &console_input_fd_ops
, &console_input
->obj
,
1308 FILE_SYNCHRONOUS_IO_NONALERT
);
1309 if (!console_input
->fd
)
1311 release_object( console_input
);
1314 return &console_input
->obj
;
1317 if (name
->len
== sizeof(outputW
) && !memcmp( name
->str
, outputW
, name
->len
))
1319 struct console_output
*console_output
;
1321 if (!(console_output
= alloc_object( &console_output_ops
))) return NULL
;
1322 console_output
->fd
= alloc_pseudo_fd( &console_output_fd_ops
, &console_output
->obj
,
1323 FILE_SYNCHRONOUS_IO_NONALERT
);
1324 if (!console_output
->fd
)
1326 release_object( console_output
);
1329 return &console_output
->obj
;
1332 if (name
->len
== sizeof(screen_bufferW
) && !memcmp( name
->str
, screen_bufferW
, name
->len
))
1334 if (!current
->process
->console
)
1336 set_error( STATUS_INVALID_HANDLE
);
1340 return create_screen_buffer( current
->process
->console
);
1343 if (name
->len
== sizeof(serverW
) && !memcmp( name
->str
, serverW
, name
->len
))
1346 return create_console_server();
1349 if (name
->len
== sizeof(connectionW
) && !memcmp( name
->str
, connectionW
, name
->len
))
1352 return create_console_connection( NULL
);
1358 static struct object
*console_device_open_file( struct object
*obj
, unsigned int access
,
1359 unsigned int sharing
, unsigned int options
)
1362 access
= default_map_access( obj
, access
);
1363 is_output
= access
& FILE_WRITE_DATA
;
1364 if (!current
->process
->console
|| (is_output
&& !current
->process
->console
))
1366 set_error( STATUS_INVALID_HANDLE
);
1369 if (is_output
&& (access
& FILE_READ_DATA
))
1371 set_error( STATUS_INVALID_PARAMETER
);
1374 return is_output
? grab_object( current
->process
->console
->active
) : grab_object( current
->process
->console
);
1377 static void console_input_dump( struct object
*obj
, int verbose
)
1379 fputs( "console Input device\n", stderr
);
1382 static int console_input_add_queue( struct object
*obj
, struct wait_queue_entry
*entry
)
1384 if (!current
->process
->console
)
1386 set_error( STATUS_ACCESS_DENIED
);
1389 return console_add_queue( ¤t
->process
->console
->obj
, entry
);
1392 static struct fd
*console_input_get_fd( struct object
*obj
)
1394 struct console_input
*console_input
= (struct console_input
*)obj
;
1395 assert( obj
->ops
== &console_input_ops
);
1396 return (struct fd
*)grab_object( console_input
->fd
);
1399 static struct object
*console_input_open_file( struct object
*obj
, unsigned int access
,
1400 unsigned int sharing
, unsigned int options
)
1402 return grab_object( obj
);
1405 static void console_input_destroy( struct object
*obj
)
1407 struct console_input
*console_input
= (struct console_input
*)obj
;
1409 assert( obj
->ops
== &console_input_ops
);
1410 if (console_input
->fd
) release_object( console_input
->fd
);
1413 static void console_input_ioctl( struct fd
*fd
, ioctl_code_t code
, struct async
*async
)
1415 struct console
*console
= current
->process
->console
;
1419 set_error( STATUS_INVALID_HANDLE
);
1422 console_ioctl( console
->fd
, code
, async
);
1425 static void console_input_read( struct fd
*fd
, struct async
*async
, file_pos_t pos
)
1427 struct console
*console
= current
->process
->console
;
1431 set_error( STATUS_INVALID_HANDLE
);
1434 console_read( console
->fd
, async
, pos
);
1437 static void console_input_flush( struct fd
*fd
, struct async
*async
)
1439 struct console
*console
= current
->process
->console
;
1443 set_error( STATUS_INVALID_HANDLE
);
1446 console_flush( console
->fd
, async
);
1449 static void console_output_dump( struct object
*obj
, int verbose
)
1451 fputs( "console Output device\n", stderr
);
1454 static int console_output_add_queue( struct object
*obj
, struct wait_queue_entry
*entry
)
1456 if (!current
->process
->console
|| !current
->process
->console
->active
)
1458 set_error( STATUS_ACCESS_DENIED
);
1461 return console_add_queue( ¤t
->process
->console
->obj
, entry
);
1464 static struct fd
*console_output_get_fd( struct object
*obj
)
1466 struct console_output
*console_output
= (struct console_output
*)obj
;
1467 assert( obj
->ops
== &console_output_ops
);
1468 return (struct fd
*)grab_object( console_output
->fd
);
1471 static struct object
*console_output_open_file( struct object
*obj
, unsigned int access
,
1472 unsigned int sharing
, unsigned int options
)
1474 return grab_object( obj
);
1477 static void console_output_destroy( struct object
*obj
)
1479 struct console_output
*console_output
= (struct console_output
*)obj
;
1481 assert( obj
->ops
== &console_output_ops
);
1482 if (console_output
->fd
) release_object( console_output
->fd
);
1485 static void console_output_ioctl( struct fd
*fd
, ioctl_code_t code
, struct async
*async
)
1487 struct console
*console
= current
->process
->console
;
1489 if (!console
|| !console
->active
)
1491 set_error( STATUS_INVALID_HANDLE
);
1494 screen_buffer_ioctl( console
->active
->fd
, code
, async
);
1497 static void console_output_write( struct fd
*fd
, struct async
*async
, file_pos_t pos
)
1499 struct console
*console
= current
->process
->console
;
1501 if (!console
|| !console
->active
)
1503 set_error( STATUS_INVALID_HANDLE
);
1506 screen_buffer_write( console
->active
->fd
, async
, pos
);
1509 struct object
*create_console_device( struct object
*root
, const struct unicode_str
*name
,
1510 unsigned int attr
, const struct security_descriptor
*sd
)
1512 return create_named_object( root
, &console_device_ops
, name
, attr
, sd
);
1515 /* retrieve the next pending console ioctl request */
1516 DECL_HANDLER(get_next_console_request
)
1518 struct console_host_ioctl
*ioctl
= NULL
, *next
;
1519 struct console_server
*server
;
1520 struct iosb
*iosb
= NULL
;
1522 server
= (struct console_server
*)get_handle_obj( current
->process
, req
->handle
, 0, &console_server_ops
);
1523 if (!server
) return;
1525 if (!server
->console
)
1527 set_error( STATUS_INVALID_HANDLE
);
1528 release_object( server
);
1532 if (!server
->console
->renderer
) server
->console
->renderer
= current
;
1534 if (!req
->signal
) server
->console
->signaled
= 0;
1535 else if (!server
->console
->signaled
)
1537 server
->console
->signaled
= 1;
1538 wake_up( &server
->console
->obj
, 0 );
1543 /* set result of current pending ioctl */
1544 if (list_empty( &server
->read_queue
))
1546 set_error( STATUS_INVALID_HANDLE
);
1547 release_object( server
);
1551 ioctl
= LIST_ENTRY( list_head( &server
->read_queue
), struct console_host_ioctl
, entry
);
1552 list_remove( &ioctl
->entry
);
1553 list_move_tail( &server
->queue
, &server
->read_queue
);
1555 else if (server
->busy
)
1557 /* set result of previous ioctl */
1558 ioctl
= LIST_ENTRY( list_head( &server
->queue
), struct console_host_ioctl
, entry
);
1559 list_remove( &ioctl
->entry
);
1564 struct async
*async
= ioctl
->async
;
1565 unsigned int status
= req
->status
;
1567 if (status
== STATUS_PENDING
) status
= STATUS_INVALID_PARAMETER
;
1570 iosb
= async_get_iosb( async
);
1571 if (iosb
->status
== STATUS_PENDING
)
1573 data_size_t out_size
= min( iosb
->out_size
, get_req_data_size() );
1574 data_size_t result
= ioctl
->code
== IOCTL_CONDRV_WRITE_FILE
? iosb
->in_size
: out_size
;
1575 async_request_complete_alloc( async
, status
, result
, out_size
, get_req_data() );
1578 release_object( async
);
1581 if (iosb
) release_object( iosb
);
1585 release_object( server
);
1591 /* if we have a blocking read ioctl in queue head and previous blocking read is still waiting,
1592 * move it to read queue for execution after current read is complete. move all blocking
1593 * ioctl at the same time to preserve their order. */
1594 if (!list_empty( &server
->queue
) && !list_empty( &server
->read_queue
))
1596 ioctl
= LIST_ENTRY( list_head( &server
->queue
), struct console_host_ioctl
, entry
);
1597 if (is_blocking_read_ioctl( ioctl
->code
))
1599 LIST_FOR_EACH_ENTRY_SAFE( ioctl
, next
, &server
->queue
, struct console_host_ioctl
, entry
)
1601 if (!is_blocking_read_ioctl( ioctl
->code
)) continue;
1602 list_remove( &ioctl
->entry
);
1603 list_add_tail( &server
->read_queue
, &ioctl
->entry
);
1608 /* return the next ioctl */
1609 if (!list_empty( &server
->queue
))
1611 ioctl
= LIST_ENTRY( list_head( &server
->queue
), struct console_host_ioctl
, entry
);
1612 iosb
= ioctl
->async
? async_get_iosb( ioctl
->async
) : NULL
;
1614 if (!iosb
|| get_reply_max_size() >= iosb
->in_size
)
1616 reply
->code
= ioctl
->code
;
1617 reply
->output
= ioctl
->output
;
1621 reply
->out_size
= iosb
->out_size
;
1622 set_reply_data_ptr( iosb
->in_data
, iosb
->in_size
);
1623 iosb
->in_data
= NULL
;
1626 if (is_blocking_read_ioctl( ioctl
->code
))
1628 list_remove( &ioctl
->entry
);
1629 assert( list_empty( &server
->read_queue
));
1630 list_add_tail( &server
->read_queue
, &ioctl
->entry
);
1632 else server
->busy
= 1;
1636 reply
->out_size
= iosb
->in_size
;
1637 set_error( STATUS_BUFFER_OVERFLOW
);
1639 if (iosb
) release_object( iosb
);
1643 set_error( STATUS_PENDING
);
1646 release_object( server
);