// Copyright 2014 The Rust Project Developers. See the COPYRIGHT // file at the top-level directory of this distribution and at // http://rust-lang.org/COPYRIGHT. // // Licensed under the Apache License, Version 2.0 or the MIT license // , at your // option. This file may not be copied, modified, or distributed // except according to those terms. // ignore-lexer-test FIXME #15877 //! Windows specific console TTY implementation //! //! This module contains the implementation of a Windows specific console TTY. //! Also converts between UTF-16 and UTF-8. Windows has very poor support for //! UTF-8 and some functions will panic. In particular ReadFile and ReadConsole //! will panic when the codepage is set to UTF-8 and a Unicode character is //! entered. //! //! FIXME //! This implementation does not account for codepoints that are split across //! multiple reads and writes. Also, this implementation does not expose a way //! to read/write UTF-16 directly. When/if Rust receives a Reader/Writer //! wrapper that performs encoding/decoding, this implementation should switch //! to working in raw UTF-16, with such a wrapper around it. use prelude::v1::*; use old_io::{self, IoError, IoResult, MemReader}; use iter::repeat; use libc::types::os::arch::extra::LPCVOID; use libc::{c_int, HANDLE, LPDWORD, DWORD, LPVOID}; use libc::{get_osfhandle, CloseHandle}; use mem; use ptr; use str::from_utf8; use super::c::{ENABLE_ECHO_INPUT, ENABLE_EXTENDED_FLAGS}; use super::c::{ENABLE_INSERT_MODE, ENABLE_LINE_INPUT}; use super::c::{ENABLE_PROCESSED_INPUT, ENABLE_QUICK_EDIT_MODE}; use super::c::CONSOLE_SCREEN_BUFFER_INFO; use super::c::{ReadConsoleW, WriteConsoleW, GetConsoleMode, SetConsoleMode}; use super::c::{GetConsoleScreenBufferInfo}; fn invalid_encoding() -> IoError { IoError { kind: old_io::InvalidInput, desc: "text was not valid unicode", detail: None, } } pub fn is_tty(fd: c_int) -> bool { let mut out: DWORD = 0; // If this function doesn't return an error, then fd is a TTY match unsafe { GetConsoleMode(get_osfhandle(fd) as HANDLE, &mut out as LPDWORD) } { 0 => false, _ => true, } } pub struct TTY { closeme: bool, handle: HANDLE, utf8: MemReader, } impl TTY { pub fn new(fd: c_int) -> IoResult { if is_tty(fd) { // If the file descriptor is one of stdin, stderr, or stdout // then it should not be closed by us let closeme = match fd { 0...2 => false, _ => true, }; let handle = unsafe { get_osfhandle(fd) as HANDLE }; Ok(TTY { handle: handle, utf8: MemReader::new(Vec::new()), closeme: closeme, }) } else { Err(IoError { kind: old_io::MismatchedFileTypeForOperation, desc: "invalid handle provided to function", detail: None, }) } } pub fn read(&mut self, buf: &mut [u8]) -> IoResult { // Read more if the buffer is empty if self.utf8.eof() { let mut utf16: Vec = repeat(0u16).take(0x1000).collect(); let mut num: DWORD = 0; match unsafe { ReadConsoleW(self.handle, utf16.as_mut_ptr() as LPVOID, utf16.len() as u32, &mut num as LPDWORD, ptr::null_mut()) } { 0 => return Err(super::last_error()), _ => (), }; utf16.truncate(num as uint); let utf8 = match String::from_utf16(&utf16) { Ok(utf8) => utf8.into_bytes(), Err(..) => return Err(invalid_encoding()), }; self.utf8 = MemReader::new(utf8); } // MemReader shouldn't error here since we just filled it Ok(self.utf8.read(buf).unwrap()) } pub fn write(&mut self, buf: &[u8]) -> IoResult<()> { let utf16 = match from_utf8(buf).ok() { Some(utf8) => { utf8.utf16_units().collect::>() } None => return Err(invalid_encoding()), }; let mut num: DWORD = 0; match unsafe { WriteConsoleW(self.handle, utf16.as_ptr() as LPCVOID, utf16.len() as u32, &mut num as LPDWORD, ptr::null_mut()) } { 0 => Err(super::last_error()), _ => Ok(()), } } pub fn set_raw(&mut self, raw: bool) -> IoResult<()> { // FIXME // Somebody needs to decide on which of these flags we want match unsafe { SetConsoleMode(self.handle, match raw { true => 0, false => ENABLE_ECHO_INPUT | ENABLE_EXTENDED_FLAGS | ENABLE_INSERT_MODE | ENABLE_LINE_INPUT | ENABLE_PROCESSED_INPUT | ENABLE_QUICK_EDIT_MODE, }) } { 0 => Err(super::last_error()), _ => Ok(()), } } pub fn get_winsize(&mut self) -> IoResult<(int, int)> { let mut info: CONSOLE_SCREEN_BUFFER_INFO = unsafe { mem::zeroed() }; match unsafe { GetConsoleScreenBufferInfo(self.handle, &mut info as *mut _) } { 0 => Err(super::last_error()), _ => Ok(((info.srWindow.Right + 1 - info.srWindow.Left) as int, (info.srWindow.Bottom + 1 - info.srWindow.Top) as int)), } } } impl Drop for TTY { fn drop(&mut self) { if self.closeme { // Nobody cares about the return value let _ = unsafe { CloseHandle(self.handle) }; } } }