summaryrefslogtreecommitdiffstats
path: root/winsup/cygwin/fhandler_console.cc
diff options
context:
space:
mode:
Diffstat (limited to 'winsup/cygwin/fhandler_console.cc')
-rw-r--r--winsup/cygwin/fhandler_console.cc158
1 files changed, 111 insertions, 47 deletions
diff --git a/winsup/cygwin/fhandler_console.cc b/winsup/cygwin/fhandler_console.cc
index 55e9b3ee8..f8a4140d6 100644
--- a/winsup/cygwin/fhandler_console.cc
+++ b/winsup/cygwin/fhandler_console.cc
@@ -78,64 +78,127 @@ beep ()
MessageBeep (MB_OK);
}
-/* Allocate and initialize the shared record for the current console.
- Returns a pointer to shared_console_info. */
-tty_min *
-fhandler_console::get_tty_stuff (int flags = 0)
+console_state *
+open_shared_console (HWND hw, HANDLE& h, shared_locations& m)
{
- if (dev_state)
- return &shared_console_info->tty_min_state;
-
- shared_console_info =
- (console_state *) open_shared (NULL, 0, cygheap->console_h,
- sizeof (*shared_console_info),
- SH_SHARED_CONSOLE);
- dev_state = &shared_console_info->dev_state;
+ wchar_t namebuf[(sizeof "XXXXXXXXXXXXXXXXXX-consNNNNNNNNNN")];
+ __small_swprintf (namebuf, L"%S-cons%p", &installation_key, hw);
+ h = NULL;
+ return (console_state *) open_shared (namebuf, 0, h,
+ sizeof (*shared_console_info),
+ &m);
+}
+class console_unit
+{
+ int n;
+ unsigned long bitmask;
+ HWND me;
+
+public:
+ operator int () const {return n;}
+ console_unit (HWND);
+ friend BOOL CALLBACK enum_windows (HWND, LPARAM);
+};
- ProtectHandleINH (cygheap->console_h);
- if (!shared_console_info->tty_min_state.ntty)
+BOOL CALLBACK
+enum_windows (HWND hw, LPARAM lp)
+{
+ console_unit *this1 = (console_unit *) lp;
+ if (hw == this1->me)
+ return TRUE;
+ shared_locations m = SH_JUSTOPEN;
+ HANDLE h;
+ console_state *cs;
+ if ((cs = open_shared_console (hw, h, m)))
{
- shared_console_info->tty_min_state.setntty (TTY_CONSOLE);
+ this1->bitmask ^= 1 << cs->tty_min_state.getntty ();
+ UnmapViewOfFile ((void *) cs);
+ CloseHandle (h);
+ }
+ return TRUE;
+}
- dev_state->scroll_region.Bottom = -1;
- dev_state->dwLastCursorPosition.X = -1;
- dev_state->dwLastCursorPosition.Y = -1;
- dev_state->dwLastMousePosition.X = -1;
- dev_state->dwLastMousePosition.Y = -1;
- dev_state->dwLastButtonState = 0; /* none pressed */
- dev_state->last_button_code = 3; /* released */
- dev_state->underline_color = FOREGROUND_GREEN | FOREGROUND_BLUE;
- dev_state->dim_color = FOREGROUND_RED | FOREGROUND_GREEN | FOREGROUND_BLUE;
- dev_state->meta_mask = LEFT_ALT_PRESSED;
- /* Set the mask that determines if an input keystroke is modified by
- META. We set this based on the keyboard layout language loaded
- for the current thread. The left <ALT> key always generates
- META, but the right <ALT> key only generates META if we are using
- an English keyboard because many "international" keyboards
- replace common shell symbols ('[', '{', etc.) with accented
- language-specific characters (umlaut, accent grave, etc.). On
- these keyboards right <ALT> (called AltGr) is used to produce the
- shell symbols and should not be interpreted as META. */
- if (PRIMARYLANGID (LOWORD (GetKeyboardLayout (0))) == LANG_ENGLISH)
- dev_state->meta_mask |= RIGHT_ALT_PRESSED;
- dev_state->set_default_attr ();
- dev_state->backspace_keycode = CERASE;
- shared_console_info->tty_min_state.sethwnd ((HWND) INVALID_HANDLE_VALUE);
+console_unit::console_unit (HWND me0):
+ bitmask (0xffffffff), me (me0)
+{
+ EnumWindows (enum_windows, (LPARAM) this);
+ n = (_minor_t) ffs (bitmask) - 1;
+ if (n < 0)
+ api_fatal ("console device allocation failure - too many consoles in use, max consoles is 32");
+}
+
+
+bool
+fhandler_console::set_unit ()
+{
+ bool need_initializing;
+ if (shared_console_info)
+ need_initializing = false;
+ else
+ {
+ HWND me = GetConsoleWindow ();
+ shared_locations m = SH_JUSTCREATE;
+ shared_console_info = open_shared_console (me, cygheap->console_h, m);
+ ProtectHandleINH (cygheap->console_h);
+ if ((need_initializing = m != SH_JUSTOPEN))
+ {
+ lock_ttys here;
+ shared_console_info->tty_min_state.setntty (DEV_CONS_MAJOR, console_unit (me));
+ }
}
+ dev_state = &shared_console_info->dev_state;
+ return need_initializing;
+}
+
+/* Allocate and initialize the shared record for the current console.
+ Returns a pointer to shared_console_info. */
+tty_min *
+fhandler_console::get_tty_stuff ()
+{
+ if (set_unit ())
+ {
+
+ dev_state->scroll_region.Bottom = -1;
+ dev_state->dwLastCursorPosition.X = -1;
+ dev_state->dwLastCursorPosition.Y = -1;
+ dev_state->dwLastMousePosition.X = -1;
+ dev_state->dwLastMousePosition.Y = -1;
+ dev_state->dwLastButtonState = 0; /* none pressed */
+ dev_state->last_button_code = 3; /* released */
+ dev_state->underline_color = FOREGROUND_GREEN | FOREGROUND_BLUE;
+ dev_state->dim_color = FOREGROUND_RED | FOREGROUND_GREEN | FOREGROUND_BLUE;
+ dev_state->meta_mask = LEFT_ALT_PRESSED;
+ /* Set the mask that determines if an input keystroke is modified by
+ META. We set this based on the keyboard layout language loaded
+ for the current thread. The left <ALT> key always generates
+ META, but the right <ALT> key only generates META if we are using
+ an English keyboard because many "international" keyboards
+ replace common shell symbols ('[', '{', etc.) with accented
+ language-specific characters (umlaut, accent grave, etc.). On
+ these keyboards right <ALT> (called AltGr) is used to produce the
+ shell symbols and should not be interpreted as META. */
+ if (PRIMARYLANGID (LOWORD (GetKeyboardLayout (0))) == LANG_ENGLISH)
+ dev_state->meta_mask |= RIGHT_ALT_PRESSED;
+ dev_state->set_default_attr ();
+ dev_state->backspace_keycode = CERASE;
+ shared_console_info->tty_min_state.sethwnd ((HWND) INVALID_HANDLE_VALUE);
+ }
+
return &shared_console_info->tty_min_state;
}
/* Return the tty structure associated with a given tty number. If the
tty number is < 0, just return a dummy record. */
tty_min *
-tty_list::get_tty (int n)
+tty_list::get_cttyp ()
{
static tty_min nada;
- if (n == TTY_CONSOLE)
+ _dev_t n = myself->ctty;
+ if (iscons_dev (n))
return &shared_console_info->tty_min_state;
- else if (n >= 0)
- return &cygwin_shared->tty.ttys[n];
+ else if (n > 0)
+ return &ttys[device::minor (n)];
else
return &nada;
}
@@ -695,7 +758,7 @@ fhandler_console::open (int flags, mode_t)
{
HANDLE h;
- tcinit (get_tty_stuff (flags), false);
+ tcinit (get_tty_stuff (), false);
set_io_handle (NULL);
set_output_handle (NULL);
@@ -847,7 +910,7 @@ fhandler_console::output_tcsetattr (int, struct termios const *t)
DWORD flags = ENABLE_PROCESSED_OUTPUT | ENABLE_WRAP_AT_EOL_OUTPUT;
int res = SetConsoleMode (get_output_handle (), flags) ? 0 : -1;
- if (!res)
+ if (res)
__seterrno_from_win_error (GetLastError ());
syscall_printf ("%d = tcsetattr (,%x) (ENABLE FLAGS %x) (lflag %x oflag %x)",
res, t, flags, t->c_lflag, t->c_oflag);
@@ -974,7 +1037,8 @@ fhandler_console::tcgetattr (struct termios *t)
fhandler_console::fhandler_console () :
fhandler_termios ()
{
- dev ().parse (FH_CONSOLE);
+ get_tty_stuff ();
+ dev ().parse (shared_console_info->tty_min_state.getntty ());
trunc_buf.len = 0;
}
@@ -2121,7 +2185,7 @@ set_console_title (char *title)
void
fhandler_console::fixup_after_fork_exec (bool execing)
{
- get_tty_stuff ();
+ ((fhandler_console *)archetype)->tc = tc = get_tty_stuff ();
}
// #define WINSTA_ACCESS (WINSTA_READATTRIBUTES | STANDARD_RIGHTS_READ | STANDARD_RIGHTS_WRITE | WINSTA_CREATEDESKTOP | WINSTA_EXITWINDOWS)