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 {
from_kernel_termios(termios: Termios) -> Self42     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)]
to_kernel_termios(&self) -> Termios56     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 = 0x0400;
144         /// 如果设置了该标志,表示启用输入流控制。
145         const IXOFF = 0x1000;
146         /// Map Uppercase to Lowercase on Input 将大写转换为小写
147         /// 表示不区分大小写
148         const IUCLC = 0x0200;
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	= 0x00004;
180         /// Map Lowercase to Uppercase on Output 输出字符时将小写字母映射为大写字母
181         const OLCUC	= 0x00002;
182 
183         /// 与NL协同 配置换行符的处理方式
184         const NLDLY	= 0x00100;
185         const   NL0	= 0x00000;  // 不延迟换行
186         const   NL1	= 0x00100;  // 延迟换行(输出回车后等待一段时间再输出换行)
187 
188         /// 配置水平制表符的处理方式
189         const TABDLY = 0x01800;
190         const  TAB0 = 0x00000;  // 不延迟水平制表符
191         const  TAB1 = 0x00800;  // 在输出水平制表符时,延迟到下一个设置的水平制表符位置
192         const  TAB2 = 0x01000;  // 在输出水平制表符时,延迟到下一个设置的 8 的倍数的位置
193         const  TAB3 = 0x01800;  // TAB3 和 XTABS(与 TAB3 等效)保留,暂未使用
194         const XTABS = 0x01800;
195 
196         /// 配置回车符的处理方式
197         const CRDLY	= 0x00600;
198         const   CR0	= 0x00000;  // 不延迟回车
199         const   CR1	= 0x02000;  //  延迟回车(输出回车后等待一段时间再输出换行)
200         const   CR2	= 0x04000;  // CR2 和 CR3保留,暂未使用
201         const   CR3	= 0x06000;
202 
203         /// 配置换页符(form feed)的处理方式
204         const FFDLY	= 0x08000;
205         const   FF0	= 0x00000;  // 不延迟换页
206         const   FF1	= 0x08000;  // 延迟换页
207 
208         /// 配置退格符(backspace)的处理方式
209         const BSDLY	= 0x02000;
210         const   BS0	= 0x00000;  // 不延迟退格
211         const   BS1	= 0x02000;  // 延迟退格
212 
213         /// 配置垂直制表符(vertical tab)的处理方式
214         const VTDLY	= 0x04000;
215         const   VT0	= 0x00000;  // 不延迟垂直制表符
216         const   VT1	= 0x04000;  // 延迟垂直制表符
217 
218         /// 表示执行输出处理,即启用输出处理函数
219         const OPOST	= 0x01;
220         /// 表示将输出的回车符 (\r) 映射为换行符 (\n)
221         const OCRNL	= 0x08;
222         /// 表示在输出时,如果光标在第 0 列,则不输出回车符 (\r)
223         const ONOCR	= 0x10;
224         /// 表示将回车符 (\r) 映射为换行符 (\n)
225         const ONLRET	= 0x20;
226         /// 表示使用填充字符进行延迟。这个填充字符的默认值是空格。
227         const OFILL	= 0x40;
228         /// 表示使用删除字符 (DEL, \177) 作为填充字符
229         const OFDEL	= 0x80;
230     }
231 
232     /// 配置终端设备的基本特性和控制参数
233     pub struct ControlMode: u32 {
234         /// Baud Rate Mask 指定波特率的掩码
235         const CBAUD		= 0x0000100f;
236         /// Extra Baud Bits 指定更高的波特率位
237         const CBAUDEX	= 0x00001000;
238         /// Custom Baud Rate 指定自定义波特率 如果设置了 BOTHER,则通过以下位来设置自定义的波特率值
239         const BOTHER	= 0x00001000;
240 
241         /* Common CBAUD rates */
242         const     B0	= 0x00000000;	/* hang up */
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 = 0x00001001;
260         const    B115200 = 0x00001002;
261         const    B230400 = 0x00001003;
262         const    B460800 = 0x00001004;
263         const    B500000 = 0x00001005;
264         const    B576000 = 0x00001006;
265         const    B921600 = 0x00001007;
266         const   B1000000 = 0x00001008;
267         const   B1152000 = 0x00001009;
268         const   B1500000 = 0x0000100a;
269         const   B2000000 = 0x0000100b;
270         const   B2500000 = 0x0000100c;
271         const   B3000000 = 0x0000100d;
272         const   B3500000 = 0x0000100e;
273         const   B4000000 = 0x0000100f;
274 
275         /// 指定字符大小的掩码 以下位为特定字符大小
276         const CSIZE		= 0x00000030;
277         const   CS5		= 0x00000000;
278         const   CS6		= 0x00000010;
279         const   CS7		= 0x00000020;
280         const   CS8		= 0x00000030;
281 
282         /// Stop Bit Select 表示使用两个停止位;否则,表示使用一个停止位
283         const CSTOPB	= 0x00000040;
284         /// 表示启用接收器。如果未设置,则禁用接收器。
285         const CREAD		= 0x00000080;
286         /// 表示启用奇偶校验。如果未设置,则禁用奇偶校验。
287         const PARENB	= 0x00000100;
288         /// 表示启用奇校验。如果未设置,则表示启用偶校验。
289         const PARODD	= 0x00000200;
290         /// 表示在终端设备被关闭时挂断线路(执行挂断操作)
291         const HUPCL		= 0x00000400;
292         /// 表示忽略调制解调器的状态(DCD、DSR、CTS 等)
293         const CLOCAL	= 0x00000800;
294         /// 指定输入波特率的掩码
295         const CIBAUD	= 0x100f0000;
296 
297         const ADDRB = 0x20000000;
298     }
299 
300     /// 配置终端设备的本地模式(local mode)或控制输入处理的行为
301     pub struct LocalMode: u32 {
302         /// 启用中断字符(Ctrl-C、Ctrl-Z)
303         const ISIG	 = 0x00001;
304         /// 表示启用规范模式,即启用行缓冲和回显。在规范模式下,输入被缓冲,并且只有在输入回车符时才会传递给应用程序。
305         const ICANON = 0x00002;
306         /// 表示启用大写模式,即输入输出都将被转换为大写。
307         const XCASE	 = 0x00004;
308         /// 表示启用回显(显示用户输入的字符)
309         const ECHO	 = 0x00008;
310         /// 表示在回显时将擦除的字符用 backspace 和空格字符显示。
311         const ECHOE	 = 0x00010;
312         /// 表示在回显时将换行符后的字符用空格字符显示。
313         const ECHOK	 = 0x00020;
314         /// 表示在回显时将换行符显示为换行和回车符。
315         const ECHONL = 0x00040;
316         /// 表示在收到中断(Ctrl-C)和退出(Ctrl-\)字符后,不清空输入和输出缓冲区。
317         const NOFLSH = 0x00080;
318         /// 表示在后台进程尝试写入终端时,发送停止信号(Ctrl-S)
319         const TOSTOP = 0x00100;
320         /// 表示在回显时,显示控制字符为 ^ 加字符。
321         const ECHOCTL= 0x00200;
322         /// 表示在回显时显示带有 # 的换行符(为了与 echo -n 命令兼容)。
323         const ECHOPRT= 0x00400;
324         /// 表示在回显时将 KILL 字符(Ctrl-U)用空格字符显示。
325         const ECHOKE = 0x00800;
326         /// 表示输出正在被冲刷(flush),通常是由于输入/输出流的状态变化。
327         const FLUSHO = 0x01000;
328         /// 表示在规范模式下,存在需要重新打印的字符。
329         const PENDIN = 0x04000;
330         /// 表示启用实现定义的输入处理。
331         const IEXTEN = 0x08000;
332         /// 表示启用扩展的处理函数
333         const EXTPROC= 0x10000;
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