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