diff options
Diffstat (limited to 'osdep/terminal-win.c')
-rw-r--r-- | osdep/terminal-win.c | 590 |
1 files changed, 475 insertions, 115 deletions
diff --git a/osdep/terminal-win.c b/osdep/terminal-win.c index 3b3a3ea935..435354155a 100644 --- a/osdep/terminal-win.c +++ b/osdep/terminal-win.c @@ -18,14 +18,12 @@ * License along with mpv. If not, see <http://www.gnu.org/licenses/>. */ -#include "config.h" #include <fcntl.h> #include <stdio.h> #include <stdint.h> #include <string.h> #include <windows.h> #include <io.h> -#include <pthread.h> #include <assert.h> #include "common/common.h" #include "input/keycodes.h" @@ -35,12 +33,38 @@ #include "osdep/threads.h" #include "osdep/w32_keyboard.h" +// https://docs.microsoft.com/en-us/windows/console/setconsolemode +// These values are effective on Windows 10 build 16257 (August 2017) or later +#ifndef ENABLE_VIRTUAL_TERMINAL_PROCESSING + #define ENABLE_VIRTUAL_TERMINAL_PROCESSING 0x0004 +#endif +#ifndef DISABLE_NEWLINE_AUTO_RETURN + #define DISABLE_NEWLINE_AUTO_RETURN 0x0008 +#endif + +// Note: the DISABLE_NEWLINE_AUTO_RETURN docs say it enables delayed-wrap, but +// it's wrong. It does only what its names suggests - and we want it unset: +// https://github.com/microsoft/terminal/issues/4126#issuecomment-571418661 +static void attempt_native_out_vt(HANDLE hOut, DWORD basemode) +{ + DWORD vtmode = basemode | ENABLE_VIRTUAL_TERMINAL_PROCESSING; + vtmode &= ~DISABLE_NEWLINE_AUTO_RETURN; + if (!SetConsoleMode(hOut, vtmode)) + SetConsoleMode(hOut, basemode); +} + + +#define hSTDIN GetStdHandle(STD_INPUT_HANDLE) #define hSTDOUT GetStdHandle(STD_OUTPUT_HANDLE) #define hSTDERR GetStdHandle(STD_ERROR_HANDLE) #define FOREGROUND_ALL (FOREGROUND_RED | FOREGROUND_GREEN | FOREGROUND_BLUE) +#define BACKGROUND_ALL (BACKGROUND_RED | BACKGROUND_GREEN | BACKGROUND_BLUE) -static short stdoutAttrs = 0; +static bool is_console[STDERR_FILENO + 1]; +static bool is_vt[STDERR_FILENO + 1]; +static bool utf8_output; +static short stdoutAttrs = 0; // copied from the screen buffer on init static const unsigned char ansi2win32[8] = { 0, FOREGROUND_RED, @@ -51,21 +75,73 @@ static const unsigned char ansi2win32[8] = { FOREGROUND_BLUE | FOREGROUND_GREEN, FOREGROUND_BLUE | FOREGROUND_GREEN | FOREGROUND_RED, }; +static const unsigned char ansi2win32bg[8] = { + 0, + BACKGROUND_RED, + BACKGROUND_GREEN, + BACKGROUND_GREEN | BACKGROUND_RED, + BACKGROUND_BLUE, + BACKGROUND_BLUE | BACKGROUND_RED, + BACKGROUND_BLUE | BACKGROUND_GREEN, + BACKGROUND_BLUE | BACKGROUND_GREEN | BACKGROUND_RED, +}; static bool running; static HANDLE death; -static pthread_t input_thread; +static mp_thread input_thread; static struct input_ctx *input_ctx; +static bool is_native_out_vt_internal(HANDLE hOut) +{ + DWORD cmode; + return GetConsoleMode(hOut, &cmode) && + (cmode & ENABLE_VIRTUAL_TERMINAL_PROCESSING) && + !(cmode & DISABLE_NEWLINE_AUTO_RETURN); +} + +static bool is_native_out_vt(HANDLE hOut) +{ + if (hOut == hSTDOUT) + return is_vt[STDOUT_FILENO]; + if (hOut == hSTDERR) + return is_vt[STDERR_FILENO]; + return is_native_out_vt_internal(hOut); +} + void terminal_get_size(int *w, int *h) { CONSOLE_SCREEN_BUFFER_INFO cinfo; - if (GetConsoleScreenBufferInfo(GetStdHandle(STD_OUTPUT_HANDLE), &cinfo)) { - *w = cinfo.dwMaximumWindowSize.X - 1; + HANDLE hOut = hSTDOUT; + if (GetConsoleScreenBufferInfo(hOut, &cinfo)) { + *w = cinfo.dwMaximumWindowSize.X - (is_native_out_vt(hOut) ? 0 : 1); *h = cinfo.dwMaximumWindowSize.Y; } } +static bool get_font_size(int *w, int *h) +{ + CONSOLE_FONT_INFO finfo; + HANDLE hOut = hSTDOUT; + BOOL res = GetCurrentConsoleFont(hOut, FALSE, &finfo); + if (res) { + *w = finfo.dwFontSize.X; + *h = finfo.dwFontSize.Y; + } + return res; +} + +void terminal_get_size2(int *rows, int *cols, int *px_width, int *px_height) +{ + int w = 0, h = 0, fw = 0, fh = 0; + terminal_get_size(&w, &h); + if (get_font_size(&fw, &fh)) { + *px_width = fw * w; + *px_height = fh * h; + *rows = w; + *cols = h; + } +} + static bool has_input_events(HANDLE h) { DWORD num_events; @@ -83,42 +159,87 @@ static void read_input(HANDLE in) break; // Only key-down events are interesting to us - if (event.EventType != KEY_EVENT) - continue; - KEY_EVENT_RECORD *record = &event.Event.KeyEvent; - if (!record->bKeyDown) - continue; - - UINT vkey = record->wVirtualKeyCode; - bool ext = record->dwControlKeyState & ENHANCED_KEY; - - int mods = 0; - if (record->dwControlKeyState & (LEFT_ALT_PRESSED | RIGHT_ALT_PRESSED)) - mods |= MP_KEY_MODIFIER_ALT; - if (record->dwControlKeyState & (LEFT_CTRL_PRESSED | RIGHT_CTRL_PRESSED)) - mods |= MP_KEY_MODIFIER_CTRL; - if (record->dwControlKeyState & SHIFT_PRESSED) - mods |= MP_KEY_MODIFIER_SHIFT; - - int mpkey = mp_w32_vkey_to_mpkey(vkey, ext); - if (mpkey) { - mp_input_put_key(input_ctx, mpkey | mods); - } else { - // Only characters should be remaining - int c = record->uChar.UnicodeChar; - // The ctrl key always produces control characters in the console. - // Shift them back up to regular characters. - if (c > 0 && c < 0x20 && (mods & MP_KEY_MODIFIER_CTRL)) - c += (mods & MP_KEY_MODIFIER_SHIFT) ? 0x40 : 0x60; - if (c >= 0x20) - mp_input_put_key(input_ctx, c | mods); + switch (event.EventType) + { + case KEY_EVENT: { + KEY_EVENT_RECORD *record = &event.Event.KeyEvent; + if (!record->bKeyDown) + continue; + + UINT vkey = record->wVirtualKeyCode; + bool ext = record->dwControlKeyState & ENHANCED_KEY; + + int mods = 0; + if (record->dwControlKeyState & (LEFT_ALT_PRESSED | RIGHT_ALT_PRESSED)) + mods |= MP_KEY_MODIFIER_ALT; + if (record->dwControlKeyState & (LEFT_CTRL_PRESSED | RIGHT_CTRL_PRESSED)) + mods |= MP_KEY_MODIFIER_CTRL; + if (record->dwControlKeyState & SHIFT_PRESSED) + mods |= MP_KEY_MODIFIER_SHIFT; + + int mpkey = mp_w32_vkey_to_mpkey(vkey, ext); + if (mpkey) { + mp_input_put_key(input_ctx, mpkey | mods); + } else { + // Only characters should be remaining + int c = record->uChar.UnicodeChar; + // The ctrl key always produces control characters in the console. + // Shift them back up to regular characters. + if (c > 0 && c < 0x20 && (mods & MP_KEY_MODIFIER_CTRL)) + c += (mods & MP_KEY_MODIFIER_SHIFT) ? 0x40 : 0x60; + if (c >= 0x20) + mp_input_put_key(input_ctx, c | mods); + } + break; + } + case MOUSE_EVENT: { + MOUSE_EVENT_RECORD *record = &event.Event.MouseEvent; + int mods = 0; + if (record->dwControlKeyState & (LEFT_ALT_PRESSED | RIGHT_ALT_PRESSED)) + mods |= MP_KEY_MODIFIER_ALT; + if (record->dwControlKeyState & (LEFT_CTRL_PRESSED | RIGHT_CTRL_PRESSED)) + mods |= MP_KEY_MODIFIER_CTRL; + if (record->dwControlKeyState & SHIFT_PRESSED) + mods |= MP_KEY_MODIFIER_SHIFT; + + switch (record->dwEventFlags) { + case MOUSE_MOVED: { + int w = 0, h = 0; + if (get_font_size(&w, &h)) { + mp_input_set_mouse_pos(input_ctx, w * (record->dwMousePosition.X + 0.5), + h * (record->dwMousePosition.Y + 0.5)); + } + break; + } + case MOUSE_HWHEELED: { + int button = (int16_t)HIWORD(record->dwButtonState) > 0 ? MP_WHEEL_RIGHT : MP_WHEEL_LEFT; + mp_input_put_key(input_ctx, button | mods); + break; + } + case MOUSE_WHEELED: { + int button = (int16_t)HIWORD(record->dwButtonState) > 0 ? MP_WHEEL_UP : MP_WHEEL_DOWN; + mp_input_put_key(input_ctx, button | mods); + break; + } + default: { + int left_button_state = record->dwButtonState & FROM_LEFT_1ST_BUTTON_PRESSED ? + MP_KEY_STATE_DOWN : MP_KEY_STATE_UP; + mp_input_put_key(input_ctx, MP_MBTN_LEFT | mods | left_button_state); + int right_button_state = record->dwButtonState & RIGHTMOST_BUTTON_PRESSED ? + MP_KEY_STATE_DOWN : MP_KEY_STATE_UP; + mp_input_put_key(input_ctx, MP_MBTN_RIGHT | mods | right_button_state); + break; + } + } + break; + } } } } -static void *input_thread_fn(void *ptr) +static MP_THREAD_VOID input_thread_fn(void *ptr) { - mpthread_set_name("terminal"); + mp_thread_set_name("terminal/input"); HANDLE in = ptr; HANDLE stuff[2] = {in, death}; while (1) { @@ -127,7 +248,7 @@ static void *input_thread_fn(void *ptr) break; read_input(in); } - return NULL; + MP_THREAD_RETURN(); } void terminal_setup_getch(struct input_ctx *ictx) @@ -141,7 +262,7 @@ void terminal_setup_getch(struct input_ctx *ictx) death = CreateEventW(NULL, TRUE, FALSE, NULL); if (!death) return; - if (pthread_create(&input_thread, NULL, input_thread_fn, in)) { + if (mp_thread_create(&input_thread, input_thread_fn, in)) { CloseHandle(death); return; } @@ -149,14 +270,21 @@ void terminal_setup_getch(struct input_ctx *ictx) } } +DWORD tmp_buffers_key = FLS_OUT_OF_INDEXES; +struct tmp_buffers { + bstr write_console_buf; + wchar_t *write_console_wbuf; +}; + void terminal_uninit(void) { if (running) { SetEvent(death); - pthread_join(input_thread, NULL); + mp_thread_join(input_thread); input_ctx = NULL; running = false; } + FlsFree(tmp_buffers_key); } bool terminal_in_background(void) @@ -164,86 +292,266 @@ bool terminal_in_background(void) return false; } -static void write_console_text(HANDLE wstream, char *buf) +int mp_console_vfprintf(HANDLE wstream, const char *format, va_list args) { - wchar_t *out = mp_from_utf8(NULL, buf); - size_t out_len = wcslen(out); - WriteConsoleW(wstream, out, out_len, NULL, NULL); - talloc_free(out); + struct tmp_buffers *buffers = FlsGetValue(tmp_buffers_key); + bool free_buf = false; + if (!buffers) { + buffers = talloc_zero(NULL, struct tmp_buffers); + free_buf = !FlsSetValue(tmp_buffers_key, buffers); + } + + buffers->write_console_buf.len = 0; + bstr_xappend_vasprintf(buffers, &buffers->write_console_buf, format, args); + + int ret = mp_console_write(wstream, buffers->write_console_buf); + + if (free_buf) + talloc_free(buffers); + + return ret; } -// Mutates the input argument (buf), because we're evil. -void mp_write_console_ansi(HANDLE wstream, char *buf) +int mp_console_write(HANDLE wstream, bstr str) { - while (*buf) { - char *next = strchr(buf, '\033'); + struct tmp_buffers *buffers = FlsGetValue(tmp_buffers_key); + bool free_buf = false; + if (!buffers) { + buffers = talloc_zero(NULL, struct tmp_buffers); + free_buf = !FlsSetValue(tmp_buffers_key, buffers); + } + + bool vt = is_native_out_vt(wstream); + int wlen = 0; + wchar_t *pos = NULL; + if (!utf8_output || !vt) { + wlen = bstr_to_wchar(buffers, str, &buffers->write_console_wbuf); + pos = buffers->write_console_wbuf; + } + + if (vt) { + if (utf8_output) { + WriteConsoleA(wstream, str.start, str.len, NULL, NULL); + } else { + WriteConsoleW(wstream, pos, wlen, NULL, NULL); + } + goto done; + } + + while (*pos) { + wchar_t *next = wcschr(pos, '\033'); if (!next) { - write_console_text(wstream, buf); + WriteConsoleW(wstream, pos, wcslen(pos), NULL, NULL); break; } - next[0] = '\0'; // mutate input for fun and profit - write_console_text(wstream, buf); - if (next[1] != '[') { - write_console_text(wstream, "\033"); - buf = next; - continue; - } - next += 2; - // ANSI codes generally follow this syntax: - // "\033[" [ <i> (';' <i> )* ] <c> - // where <i> are integers, and <c> a single char command code. - // Also see: http://en.wikipedia.org/wiki/ANSI_escape_code#CSI_codes - int params[2] = {-1, -1}; // 'm' might be unlimited; ignore that - int num_params = 0; - while (num_params < 2) { - char *end = next; - long p = strtol(next, &end, 10); - if (end == next) + next[0] = '\0'; + WriteConsoleW(wstream, pos, wcslen(pos), NULL, NULL); + if (next[1] == '[') { + // CSI - Control Sequence Introducer + next += 2; + + // private sequences + bool priv = next[0] == '?'; + next += priv; + + // CSI codes generally follow this syntax: + // "\033[" [ <i> (';' <i> )* ] <c> + // where <i> are integers, and <c> a single char command code. + // Also see: http://en.wikipedia.org/wiki/ANSI_escape_code#CSI_codes + int params[16]; // 'm' might be unlimited; ignore that + int num_params = 0; + while (num_params < MP_ARRAY_SIZE(params)) { + wchar_t *end = next; + long p = wcstol(next, &end, 10); + if (end == next) + break; + next = end; + params[num_params++] = p; + if (next[0] != ';' || !next[0]) + break; + next += 1; + } + wchar_t code = next[0]; + if (code) + next += 1; + CONSOLE_SCREEN_BUFFER_INFO info; + GetConsoleScreenBufferInfo(wstream, &info); + switch (code) { + case 'K': { // erase line + COORD cursor_pos = info.dwCursorPosition; + COORD at = cursor_pos; + int len; + switch (num_params ? params[0] : 0) { + case 1: + len = at.X; + at.X = 0; + break; + case 2: + len = info.dwSize.X; + at.X = 0; + break; + case 0: + default: + len = info.dwSize.X - at.X; + } + FillConsoleOutputCharacterW(wstream, L' ', len, at, &(DWORD){0}); + SetConsoleCursorPosition(wstream, cursor_pos); break; - next = end; - params[num_params++] = p; - if (next[0] != ';' || !next[0]) + } + case 'B': { // cursor down + info.dwCursorPosition.Y += !num_params ? 1 : params[0]; + SetConsoleCursorPosition(wstream, info.dwCursorPosition); break; - next += 1; - } - char code = next[0]; - if (code) - next += 1; - CONSOLE_SCREEN_BUFFER_INFO info; - GetConsoleScreenBufferInfo(wstream, &info); - switch (code) { - case 'K': { // erase to end of line - COORD at = info.dwCursorPosition; - int len = info.dwSize.X - at.X; - FillConsoleOutputCharacterW(wstream, ' ', len, at, &(DWORD){0}); - SetConsoleCursorPosition(wstream, at); - break; - } - case 'A': { // cursor up - info.dwCursorPosition.Y -= 1; - SetConsoleCursorPosition(wstream, info.dwCursorPosition); - break; - } - case 'm': { // "SGR" - for (int n = 0; n < num_params; n++) { - int p = params[n]; - if (p == 0) { - info.wAttributes = stdoutAttrs; - SetConsoleTextAttribute(wstream, info.wAttributes); - } else if (p == 1) { - info.wAttributes |= FOREGROUND_INTENSITY; - SetConsoleTextAttribute(wstream, info.wAttributes); - } else if (p >= 30 && p < 38) { - info.wAttributes &= ~FOREGROUND_ALL; - info.wAttributes |= ansi2win32[p - 30]; - SetConsoleTextAttribute(wstream, info.wAttributes); + } + case 'A': { // cursor up + info.dwCursorPosition.Y -= !num_params ? 1 : params[0]; + SetConsoleCursorPosition(wstream, info.dwCursorPosition); + break; + } + case 'J': { + // Only full screen clear is supported + if (!num_params || params[0] != 2) + break; + + COORD top_left = {0, 0}; + FillConsoleOutputCharacterW(wstream, L' ', info.dwSize.X * info.dwSize.Y, + top_left, &(DWORD){0}); + SetConsoleCursorPosition(wstream, top_left); + break; + } + case 'f': { + if (num_params != 2) + break; + SetConsoleCursorPosition(wstream, (COORD){params[0], params[1]}); + break; + } + case 'l': { + if (!priv || !num_params) + break; + + switch (params[0]) { + case 25:; // hide the cursor + CONSOLE_CURSOR_INFO cursor_info; + if (!GetConsoleCursorInfo(wstream, &cursor_info)) + break; + cursor_info.bVisible = FALSE; + SetConsoleCursorInfo(wstream, &cursor_info); + break; } + break; } - break; - } + case 'h': { + if (!priv || !num_params) + break; + + switch (params[0]) { + case 25:; // show the cursor + CONSOLE_CURSOR_INFO cursor_info; + if (!GetConsoleCursorInfo(wstream, &cursor_info)) + break; + cursor_info.bVisible = TRUE; + SetConsoleCursorInfo(wstream, &cursor_info); + break; + } + break; + } + case 'm': { // "SGR" + short attr = info.wAttributes; + if (num_params == 0) // reset + params[num_params++] = 0; + + // we don't emulate italic, reverse/underline don't always work + for (int n = 0; n < num_params; n++) { + int p = params[n]; + if (p == 0) { + attr = stdoutAttrs; + } else if (p == 1) { + attr |= FOREGROUND_INTENSITY; + } else if (p == 22) { + attr &= ~FOREGROUND_INTENSITY; + } else if (p == 4) { + attr |= COMMON_LVB_UNDERSCORE; + } else if (p == 24) { + attr &= ~COMMON_LVB_UNDERSCORE; + } else if (p == 7) { + attr |= COMMON_LVB_REVERSE_VIDEO; + } else if (p == 27) { + attr &= ~COMMON_LVB_REVERSE_VIDEO; + } else if (p >= 30 && p <= 37) { + attr &= ~FOREGROUND_ALL; + attr |= ansi2win32[p - 30]; + } else if (p == 39) { + attr &= ~FOREGROUND_ALL; + attr |= stdoutAttrs & FOREGROUND_ALL; + } else if (p >= 40 && p <= 47) { + attr &= ~BACKGROUND_ALL; + attr |= ansi2win32bg[p - 40]; + } else if (p == 49) { + attr &= ~BACKGROUND_ALL; + attr |= stdoutAttrs & BACKGROUND_ALL; + } else if (p == 38 || p == 48) { // ignore and skip sub-values + // 256 colors: <38/48>;5;N true colors: <38/48>;2;R;G;B + if (n+1 < num_params) { + n += params[n+1] == 5 ? 2 + : params[n+1] == 2 ? 4 + : num_params; /* unrecognized -> the rest */ + } + } + } + + if (attr != info.wAttributes) + SetConsoleTextAttribute(wstream, attr); + break; + } + } + } else if (next[1] == ']') { + // OSC - Operating System Commands + next += 2; + + // OSC sequences generally follow this syntax: + // "\033]" <command> ST + // Where <command> is a string command + wchar_t *cmd = next; + while (next[0]) { + // BEL can be used instead of ST in xterm + if (next[0] == '\007' || next[0] == 0x9c) { + next[0] = '\0'; + next += 1; + break; + } + if (next[0] == '\033' && next[1] == '\\') { + next[0] = '\0'; + next += 2; + break; + } + next += 1; + } + + // Handle xterm-style OSC commands + if (cmd[0] && cmd[1] == ';') { + wchar_t code = cmd[0]; + wchar_t *param = cmd + 2; + + switch (code) { + case '0': // Change Icon Name and Window Title + case '2': // Change Window Title + SetConsoleTitleW(param); + break; + } + } + } else { + WriteConsoleW(wstream, L"\033", 1, NULL, NULL); } - buf = next; + pos = next; } + +done:; + int ret = buffers->write_console_buf.len; + + if (free_buf) + talloc_free(buffers); + + return ret; } static bool is_a_console(HANDLE h) @@ -251,12 +559,39 @@ static bool is_a_console(HANDLE h) return GetConsoleMode(h, &(DWORD){0}); } +bool mp_check_console(void *handle) +{ + if (handle == hSTDIN) + return is_console[STDIN_FILENO]; + if (handle == hSTDOUT) + return is_console[STDOUT_FILENO]; + if (handle == hSTDERR) + return is_console[STDERR_FILENO]; + return is_a_console(handle); +} + static void reopen_console_handle(DWORD std, int fd, FILE *stream) { - if (is_a_console(GetStdHandle(std))) { - freopen("CONOUT$", "wt", stream); - dup2(fileno(stream), fd); - setvbuf(stream, NULL, _IONBF, 0); + HANDLE handle = GetStdHandle(std); + if (is_a_console(handle)) { + if (fd == 0) { + freopen("CONIN$", "rt", stream); + } else { + freopen("CONOUT$", "wt", stream); + } + + // Set the low-level FD to the new handle value, since mp_subprocess2 + // callers might rely on low-level FDs being set. Note, with this + // method, fileno(stdin) != STDIN_FILENO, but that shouldn't matter. + int unbound_fd = -1; + if (fd == 0) { + unbound_fd = _open_osfhandle((intptr_t)handle, _O_RDONLY); + } else { + unbound_fd = _open_osfhandle((intptr_t)handle, _O_WRONLY); + } + // dup2 will duplicate the underlying handle. Don't close unbound_fd, + // since that will close the original handle. + dup2(unbound_fd, fd); } } @@ -277,22 +612,47 @@ bool terminal_try_attach(void) if (!AttachConsole(ATTACH_PARENT_PROCESS)) return false; - // We have a console window. Redirect output streams to that console's - // low-level handles, so things that use printf directly work later on. + // We have a console window. Redirect input/output streams to that console's + // low-level handles, so things that use stdio work later on. + reopen_console_handle(STD_INPUT_HANDLE, STDIN_FILENO, stdin); reopen_console_handle(STD_OUTPUT_HANDLE, STDOUT_FILENO, stdout); reopen_console_handle(STD_ERROR_HANDLE, STDERR_FILENO, stderr); return true; } +void terminal_set_mouse_input(bool enable) +{ + DWORD cmode; + HANDLE in = hSTDIN; + if (GetConsoleMode(in, &cmode)) { + cmode = enable ? cmode | ENABLE_MOUSE_INPUT + : cmode & (~ENABLE_MOUSE_INPUT); + SetConsoleMode(in, cmode); + } +} + void terminal_init(void) { CONSOLE_SCREEN_BUFFER_INFO cinfo; DWORD cmode = 0; GetConsoleMode(hSTDOUT, &cmode); cmode |= (ENABLE_PROCESSED_OUTPUT | ENABLE_WRAP_AT_EOL_OUTPUT); - SetConsoleMode(hSTDOUT, cmode); - SetConsoleMode(hSTDERR, cmode); + attempt_native_out_vt(hSTDOUT, cmode); + attempt_native_out_vt(hSTDERR, cmode); + + // Init for mp_check_console(), this never changes during runtime + is_console[STDIN_FILENO] = is_a_console(hSTDIN); + is_console[STDOUT_FILENO] = is_a_console(hSTDOUT); + is_console[STDERR_FILENO] = is_a_console(hSTDERR); + + // Init for is_native_out_vt(), this is never disabled/changed during runtime + is_vt[STDOUT_FILENO] = is_native_out_vt_internal(hSTDOUT); + is_vt[STDERR_FILENO] = is_native_out_vt_internal(hSTDERR); + GetConsoleScreenBufferInfo(hSTDOUT, &cinfo); stdoutAttrs = cinfo.wAttributes; + + tmp_buffers_key = FlsAlloc((PFLS_CALLBACK_FUNCTION)talloc_free); + utf8_output = SetConsoleOutputCP(CP_UTF8); } |