1 use super::tty_ldisc::LineDisciplineType; 2 3 /// ## 窗口大小 4 #[repr(C)] 5 #[derive(Debug, Default, Clone, Copy)] 6 pub struct WindowSize { 7 /// 行 8 pub row: u16, 9 /// 列 10 pub col: u16, 11 /// x方向像素数 12 pub xpixel: u16, 13 /// y方向像素数 14 pub ypixel: u16, 15 } 16 17 #[derive(Debug, Clone, Copy)] 18 pub struct Termios { 19 pub input_mode: InputMode, 20 pub output_mode: OutputMode, 21 pub control_mode: ControlMode, 22 pub local_mode: LocalMode, 23 pub control_characters: [u8; CONTORL_CHARACTER_NUM], 24 pub line: LineDisciplineType, 25 pub input_speed: u32, 26 pub output_speed: u32, 27 } 28 29 #[derive(Clone, Copy, Default)] 30 pub struct PosixTermios { 31 pub c_iflag: u32, 32 pub c_oflag: u32, 33 pub c_cflag: u32, 34 pub c_lflag: u32, 35 pub c_cc: [u8; CONTORL_CHARACTER_NUM], 36 pub c_line: u8, 37 pub c_ispeed: u32, 38 pub c_ospeed: u32, 39 } 40 41 impl PosixTermios { 42 pub fn from_kernel_termios(termios: Termios) -> Self { 43 Self { 44 c_iflag: termios.input_mode.bits, 45 c_oflag: termios.output_mode.bits, 46 c_cflag: termios.control_mode.bits, 47 c_lflag: termios.local_mode.bits, 48 c_cc: termios.control_characters.clone(), 49 c_line: termios.line as u8, 50 c_ispeed: termios.input_speed, 51 c_ospeed: termios.output_speed, 52 } 53 } 54 55 #[allow(dead_code)] 56 pub fn to_kernel_termios(&self) -> Termios { 57 // TODO:这里没有考虑非规范模式 58 Termios { 59 input_mode: InputMode::from_bits_truncate(self.c_iflag), 60 output_mode: OutputMode::from_bits_truncate(self.c_oflag), 61 control_mode: ControlMode::from_bits_truncate(self.c_cflag), 62 local_mode: LocalMode::from_bits_truncate(self.c_lflag), 63 control_characters: self.c_cc.clone(), 64 line: LineDisciplineType::from_line(self.c_line), 65 input_speed: self.c_ispeed, 66 output_speed: self.c_ospeed, 67 } 68 } 69 } 70 71 pub const INIT_CONTORL_CHARACTERS: [u8; CONTORL_CHARACTER_NUM] = [ 72 b'C' - 0x40, // VINTR 73 b'\\' - 0x40, // VQUIT 74 0o177, // VERASE 75 b'U' - 0x40, // VKILL 76 b'D' - 0x40, // VEOF 77 1, // VMIN 78 0, // VEOL 79 0, // VTIME 80 0, // VEOL2 81 0, // VSWTC 82 b'W' - 0x40, // VWERASE 83 b'R' - 0x40, // VREPRINT 84 b'Z' - 0x40, // VSUSP 85 b'Q' - 0x40, // VSTART 86 b'S' - 0x40, // VSTOP 87 b'V' - 0x40, // VLNEXT 88 b'O' - 0x40, // VDISCARD 89 0, 90 0, 91 ]; 92 93 // pub const INIT_CONTORL_CHARACTERS: [u8; CONTORL_CHARACTER_NUM] = { 94 // let mut chs: [u8; CONTORL_CHARACTER_NUM] = Default::default(); 95 // chs[ControlCharIndex::VINTR] = b'C' - 0x40; 96 // chs[ControlCharIndex::VQUIT] = b'\\' - 0x40; 97 // chs[ControlCharIndex::VERASE] = 0o177; 98 // chs[ControlCharIndex::VKILL] = b'U' - 0x40; 99 // chs[ControlCharIndex::VEOF] = b'D' - 0x40; 100 // chs[ControlCharIndex::VSTART] = b'Q' - 0x40; 101 // chs[ControlCharIndex::VSTOP] = b'S' - 0x40; 102 // chs[ControlCharIndex::VSUSP] = b'Z' - 0x40; 103 // chs[ControlCharIndex::VREPRINT] = b'R' - 0x40; 104 // chs[ControlCharIndex::VDISCARD] = b'O' - 0x40; 105 // chs[ControlCharIndex::VWERASE] = b'W' - 0x40; 106 // chs[ControlCharIndex::VLNEXT] = b'V' - 0x40; 107 // // chs[ContorlCharIndex::VDSUSP] = b'Y' - 0x40; 108 // chs[ControlCharIndex::VMIN] = 1; 109 // return chs; 110 // }; 111 112 lazy_static! { 113 pub static ref TTY_STD_TERMIOS: Termios = { 114 Termios { 115 input_mode: InputMode::ICRNL | InputMode::IXON, 116 output_mode: OutputMode::OPOST | OutputMode::ONLCR, 117 control_mode: ControlMode::B38400 118 | ControlMode::CREAD 119 | ControlMode::HUPCL 120 | ControlMode::CS8, 121 local_mode: LocalMode::ISIG 122 | LocalMode::ICANON 123 | LocalMode::ECHO 124 | LocalMode::ECHOE 125 | LocalMode::ECHOK 126 | LocalMode::ECHOCTL 127 | LocalMode::ECHOKE 128 | LocalMode::IEXTEN, 129 control_characters: INIT_CONTORL_CHARACTERS.clone(), 130 line: LineDisciplineType::NTty, 131 input_speed: 38400, 132 output_speed: 38400, 133 } 134 }; 135 } 136 137 pub const CONTORL_CHARACTER_NUM: usize = 19; 138 139 bitflags! { 140 /// termios输入特性 141 pub struct InputMode: u32 { 142 /// 如果设置了该标志,表示启用软件流控制。 143 const IXON = 0x0200; 144 /// 如果设置了该标志,表示启用输入流控制。 145 const IXOFF = 0x0400; 146 /// Map Uppercase to Lowercase on Input 将大写转换为小写 147 /// 表示不区分大小写 148 const IUCLC = 0x1000; 149 /// 如果设置了该标志,表示当输入队列满时,产生一个响铃信号。 150 const IMAXBEL = 0x2000; 151 /// 如果设置了该标志,表示输入数据被视为 UTF-8 编码。 152 const IUTF8 = 0x4000; 153 154 /// 忽略中断信号 155 const IGNBRK = 0x001; 156 /// 检测到中断信号时生成中断(产生中断信号) 157 const BRKINT = 0x002; 158 /// 忽略具有奇偶校验错误的字符 159 const IGNPAR = 0x004; 160 /// 在检测到奇偶校验错误或帧错误时,将字符以 \377 标记 161 const PARMRK = 0x008; 162 /// 启用输入奇偶校验检查 163 const INPCK = 0x010; 164 /// 从输入字符中剥离第 8 位,即只保留低 7 位 165 const ISTRIP = 0x020; 166 /// 表示将输入的换行符 (\n) 映射为回车符 (\r) 167 const INLCR = 0x040; 168 /// 表示忽略回车符 (\r) 169 const IGNCR = 0x080; 170 /// 表示将输入的回车符 (\r) 映射为换行符 (\n) 171 const ICRNL = 0x100; 172 /// 表示在输入被停止(Ctrl-S)后,任何字符的输入都将重新启动输入 173 const IXANY = 0x800; 174 } 175 176 /// termios输出特性 177 pub struct OutputMode: u32 { 178 /// 在输出时将换行符替换\r\n 179 const ONLCR = 0x00002; 180 /// Map Lowercase to Uppercase on Output 输出字符时将小写字母映射为大写字母 181 const OLCUC = 0x00004; 182 183 /// 与NL协同 配置换行符的处理方式 184 const NLDLY = 0x00300; 185 const NL0 = 0x00000; // 不延迟换行 186 const NL1 = 0x00100; // 延迟换行(输出回车后等待一段时间再输出换行) 187 const NL2 = 0x00200; // NL2 和 NL3保留,暂未使用 188 const NL3 = 0x00300; 189 190 /// 配置水平制表符的处理方式 191 const TABDLY = 0x00c00; 192 const TAB0 = 0x00000; // 不延迟水平制表符 193 const TAB1 = 0x00400; // 在输出水平制表符时,延迟到下一个设置的水平制表符位置 194 const TAB2 = 0x00800; // 在输出水平制表符时,延迟到下一个设置的 8 的倍数的位置 195 const TAB3 = 0x00c00; // TAB3 和 XTABS(与 TAB3 等效)保留,暂未使用 196 const XTABS = 0x00c00; 197 198 /// 配置回车符的处理方式 199 const CRDLY = 0x03000; 200 const CR0 = 0x00000; // 不延迟回车 201 const CR1 = 0x01000; // 延迟回车(输出回车后等待一段时间再输出换行) 202 const CR2 = 0x02000; // CR2 和 CR3保留,暂未使用 203 const CR3 = 0x03000; 204 205 /// 配置换页符(form feed)的处理方式 206 const FFDLY = 0x04000; 207 const FF0 = 0x00000; // 不延迟换页 208 const FF1 = 0x04000; // 延迟换页 209 210 /// 配置退格符(backspace)的处理方式 211 const BSDLY = 0x08000; 212 const BS0 = 0x00000; // 不延迟退格 213 const BS1 = 0x08000; // 延迟退格 214 215 /// 配置垂直制表符(vertical tab)的处理方式 216 const VTDLY = 0x10000; 217 const VT0 = 0x00000; // 不延迟垂直制表符 218 const VT1 = 0x10000; // 延迟垂直制表符 219 220 /// 表示执行输出处理,即启用输出处理函数 221 const OPOST = 0x01; 222 /// 表示将输出的回车符 (\r) 映射为换行符 (\n) 223 const OCRNL = 0x08; 224 /// 表示在输出时,如果光标在第 0 列,则不输出回车符 (\r) 225 const ONOCR = 0x10; 226 /// 表示将回车符 (\r) 映射为换行符 (\n) 227 const ONLRET = 0x20; 228 /// 表示使用填充字符进行延迟。这个填充字符的默认值是空格。 229 const OFILL = 0x40; 230 /// 表示使用删除字符 (DEL, \177) 作为填充字符 231 const OFDEL = 0x80; 232 } 233 234 /// 配置终端设备的基本特性和控制参数 235 pub struct ControlMode: u32 { 236 /// Baud Rate Mask 指定波特率的掩码 237 const CBAUD = 0x000000ff; 238 /// Extra Baud Bits 指定更高的波特率位 239 const CBAUDEX = 0x00000000; 240 /// Custom Baud Rate 指定自定义波特率 如果设置了 BOTHER,则通过以下位来设置自定义的波特率值 241 const BOTHER = 0x0000001f; 242 243 const B0 = 0x00000000; 244 const B50 = 0x00000001; 245 const B75 = 0x00000002; 246 const B110 = 0x00000003; 247 const B134 = 0x00000004; 248 const B150 = 0x00000005; 249 const B200 = 0x00000006; 250 const B300 = 0x00000007; 251 const B600 = 0x00000008; 252 const B1200 = 0x00000009; 253 const B1800 = 0x0000000a; 254 const B2400 = 0x0000000b; 255 const B4800 = 0x0000000c; 256 const B9600 = 0x0000000d; 257 const B19200 = 0x0000000e; 258 const B38400 = 0x0000000f; 259 260 const B57600 = 0x00000010; 261 const B115200 = 0x00000011; 262 const B230400 = 0x00000012; 263 const B460800 = 0x00000013; 264 const B500000 = 0x00000014; 265 const B576000 = 0x00000015; 266 const B921600 = 0x00000016; 267 const B1000000 = 0x00000017; 268 const B1152000 = 0x00000018; 269 const B1500000 = 0x00000019; 270 const B2000000 = 0x0000001a; 271 const B2500000 = 0x0000001b; 272 const B3000000 = 0x0000001c; 273 const B3500000 = 0x0000001d; 274 const B4000000 = 0x0000001e; 275 276 /// 指定字符大小的掩码 以下位为特定字符大小 277 const CSIZE = 0x00000300; 278 const CS5 = 0x00000000; 279 const CS6 = 0x00000100; 280 const CS7 = 0x00000200; 281 const CS8 = 0x00000300; 282 283 /// Stop Bit Select 表示使用两个停止位;否则,表示使用一个停止位 284 const CSTOPB = 0x00000400; 285 /// 表示启用接收器。如果未设置,则禁用接收器。 286 const CREAD = 0x00000800; 287 /// 表示启用奇偶校验。如果未设置,则禁用奇偶校验。 288 const PARENB = 0x00001000; 289 /// 表示启用奇校验。如果未设置,则表示启用偶校验。 290 const PARODD = 0x00002000; 291 /// 表示在终端设备被关闭时挂断线路(执行挂断操作) 292 const HUPCL = 0x00004000; 293 /// 表示忽略调制解调器的状态(DCD、DSR、CTS 等) 294 const CLOCAL = 0x00008000; 295 /// 指定输入波特率的掩码 296 const CIBAUD = 0x00ff0000; 297 298 const ADDRB = 0x20000000; 299 } 300 301 /// 配置终端设备的本地模式(local mode)或控制输入处理的行为 302 pub struct LocalMode: u32 { 303 /// 启用中断字符(Ctrl-C、Ctrl-Z) 304 const ISIG = 0x00000080; 305 /// 表示启用规范模式,即启用行缓冲和回显。在规范模式下,输入被缓冲,并且只有在输入回车符时才会传递给应用程序。 306 const ICANON = 0x00000100; 307 /// 表示启用大写模式,即输入输出都将被转换为大写。 308 const XCASE = 0x00004000; 309 /// 表示启用回显(显示用户输入的字符) 310 const ECHO = 0x00000008; 311 /// 表示在回显时将擦除的字符用 backspace 和空格字符显示。 312 const ECHOE = 0x00000002; 313 /// 表示在回显时将换行符后的字符用空格字符显示。 314 const ECHOK = 0x00000004; 315 /// 表示在回显时将换行符显示为换行和回车符。 316 const ECHONL = 0x00000010; 317 /// 表示在收到中断(Ctrl-C)和退出(Ctrl-\)字符后,不清空输入和输出缓冲区。 318 const NOFLSH = 0x80000000; 319 /// 表示在后台进程尝试写入终端时,发送停止信号(Ctrl-S) 320 const TOSTOP = 0x00400000; 321 /// 表示在回显时,显示控制字符为 ^ 加字符。 322 const ECHOCTL= 0x00000040; 323 /// 表示在回显时显示带有 # 的换行符(为了与 echo -n 命令兼容)。 324 const ECHOPRT= 0x00000020; 325 /// 表示在回显时将 KILL 字符(Ctrl-U)用空格字符显示。 326 const ECHOKE = 0x00000001; 327 /// 表示输出正在被冲刷(flush),通常是由于输入/输出流的状态变化。 328 const FLUSHO = 0x00800000; 329 /// 表示在规范模式下,存在需要重新打印的字符。 330 const PENDIN = 0x20000000; 331 /// 表示启用实现定义的输入处理。 332 const IEXTEN = 0x00000400; 333 /// 表示启用扩展的处理函数 334 const EXTPROC= 0x10000000; 335 } 336 337 pub struct TtySetTermiosOpt: u8 { 338 const TERMIOS_FLUSH =1; 339 const TERMIOS_WAIT =2; 340 const TERMIOS_TERMIO =4; 341 const TERMIOS_OLD =8; 342 } 343 } 344 345 /// 对应termios中控制字符的索引 346 pub struct ControlCharIndex; 347 #[allow(dead_code)] 348 impl ControlCharIndex { 349 pub const DISABLE_CHAR: u8 = b'\0'; 350 /// 中断信号 351 pub const VINTR: usize = 0; 352 /// 退出信号 353 pub const VQUIT: usize = 1; 354 /// 退格 355 pub const VERASE: usize = 2; 356 /// 终止输入信号 357 pub const VKILL: usize = 3; 358 /// 文件结束信号 \0? 359 pub const VEOF: usize = 4; 360 /// 指定非规范模式下的最小字符数 361 pub const VMIN: usize = 5; 362 /// 换行符 363 pub const VEOL: usize = 6; 364 /// 指定非规范模式下的超时时间 365 pub const VTIME: usize = 7; 366 /// 换行符 367 pub const VEOL2: usize = 8; 368 /// 未使用,保留 369 pub const VSWTC: usize = 9; 370 /// 擦除前一个单词 371 pub const VWERASE: usize = 10; 372 /// 重新打印整行 373 pub const VREPRINT: usize = 11; 374 /// 挂起信号 375 pub const VSUSP: usize = 12; 376 /// 启动输出信号 377 pub const VSTART: usize = 13; 378 /// 停止输出信号 379 pub const VSTOP: usize = 14; 380 /// 将下一个字符视为字面值,而不是特殊字符 381 pub const VLNEXT: usize = 15; 382 /// 对应于字符丢弃信号,用于丢弃当前输入的行 383 pub const VDISCARD: usize = 16; 384 } 385