540 likes | 854 Views
Terminal Control. Terminal Control may seem like an arcane subject, but …. It illustrates the relationship between devices and files. The terminal driver is an easy and fun device driver to work with. Consider the following code: #include <stdio.h> int main ( ) { int c, n = 0;
E N D
Terminal Control may seem like an arcane subject, but … It illustrates the relationship between devices and files The terminal driver is an easy and fun device driver to work with
Consider the following code: #include <stdio.h> int main ( ) { int c, n = 0; while( ( c = getchar( ) ) != ‘Q’ ) printf ( “char %3d is %c code %d\n”, n++, c, c); return 0; } Run it ..... (listchars.c)
Some Observations The program does not process any data until the Enter key is hit. The enter key generates a Carriage Return character, 13 but the program receives the character code 10, New Line.
The Terminal Driver TerminalDriver Application The Terminal driver is doing some manipulation of the data as it flows between the terminal and your program!
The Terminal Driver TerminalDriver Application attributes These manipulations are controlled by terminal attributes that can be changed by a process.
Terminal Attributes To see the current set of terminal attributes stty stty -a
Local mode flags: * icanon: enable/disable canonical mode * isig: enable/disable checking for INTR, SUSP, QUIT * echo: echo every character typed
Input mode flags: * inlcr: map/do not map NLto CR on input * igncr: ignore/do not ignore CR on input * icrnl: map/do not map CR to NL on input
Output mode flags: * onlcr: map/do not map NLto CR on output * ocrnl: map/do not map CR to NL
$ stty -a speed 9600 baud; 24 rows; 80 columns; . . . iflags: -istrp icrnl -inlcr . . . convert cr to nl - the flag is turned off man stty to see more detail
You can alter a setting with the stty command stty -echo This turns off echo mode
Terminal Modes Canonical or cooked mode The terminal driver stores incoming characters in a buffer Sends characters to the application a line at a time (when you press Enter) Handles basic editing functions (delete, backspace, …)
Terminal Modes canonical mode TerminalDriver Application buffer lines characters Process Backspace, etc
Terminal Modes Non-Canonical or cr-break mode The terminal driver does not buffer keyboard input Sends characters to the application a character at a time, as it is entered Does some character processing * Ctrl-C (interrupt) * carriage return to newline
Terminal Modes non-canonical mode TerminalDriver Application characters characters Process Ctrl-C, CR-NL
Terminal Modes Raw mode The terminal driver does not buffer keyboard input. Sends characters to the application a character at a time Does no character processing
Terminal Modes raw mode TerminalDriver Application
The Terminal Driver You control the Terminal driver by manipulating A set of terminal attributes stored in the device driver. TerminalDriver Application driver attributes tcgetattr( ) tcsetattr( )
Reading attributes from the Terminal Driver #include <termios.h> #include <unistd.h> int result = tcgetattr(int fd, struct termios* attrs); returns 0 if successful -1 if an error zero for stdin address of a termios struct
Setting attributes in the Terminal Driver #include <termios.h> #include <unistd.h> int result = tcsetattr(int fd, int when, struct termios* attrs); returns 0 if successful -1 if an error when to apply the settings zero for stdin address of a termios struct TCSANOW – immediately TCSADRAIN – after draining queued data TCSAFLUSH – after draining + flush input Note: Some shells will reset these attributes in an effort to protect you from yourself.
The termios Struct struct termios { tcflag_t c_iflag; /* input mode flags */ tcflag_t c_oflag; /* output mode flags */ tcflag_t c_cflag; /* control mode flags */ tcflag_t c_lflag; /* local mode flags */ cc_t c_cc[NCCS]; /* control characters */ speed_t c_ispeed; /* input speed */ speed_t c_ospeed; /* output speed */ };
termios Flags Each flag is a bit in one of the mode fields. All of these flags have symbolic constants. To see all of the fields in the termios structure see termios man pages. These are the ones we are interested in. c_lflag ECHO – enable echo ICANON – canonical mode ISIG – enable signals
Remember your boolean logic? Bitwise Operators & and | or ~ complement a b a & b a b a | b 1 1 1 1 0 0 0 1 0 0 0 0 1 1 1 1 0 1 0 1 1 0 0 0
To Set a Flag flag = flag | mask 0’s except for the bits to be set in the flag flag mask flag | mask 1 1 1 1 0 1 0 1 1 0 0 0 this bit got set
To Set a Flag 0 1 1 0 Flags Suppose I want to set this bit Mask What goes in the mask and what operation do I use?
To Clear a Flag flag = flag & ~mask 1’s except for the bits to be cleared in the flag flag mask ~ mask flag & ~mask 1 1 0 0 1 0 1 1 0 1 0 0 0 0 1 0 this bit got cleared
To Clear a Flag 0 1 1 0 Flags Suppose I want to clear this bit Mask What goes in the mask and what operation do I use?
To Test a Flag if ( flag == flag & mask ) set the bits in the mask that you want to test in the flag flag mask flag & mask 1 1 1 1 0 0 0 1 0 0 0 0 Remember that in C, 0 is false and non-zero is true.
To Test a Flag 0 1 1 0 Flags Suppose I want to know what this bit is Mask What goes in the mask and what operation do I use?
Examine Echo flag #include <stdio.h> #include <termios.h> #include <stdlib.h> int main ( ) { struct termios info; int rv; if ( (rv = tcgetattr( 0, &info ) ) == -1 ) { perror(“tcgetattr error”); exit ( 1 ); } if ( info.c_lflag & ECHO) printf(“ echo flag is set\n”); else printf(“ echo flag is not set\n”); return 0; } file descriptor is 0 for stdin Example test the flag echostate.c
Change Echo flag Example if ( (rv = tcgetattr( 0, &info ) ) == -1 ) { perror(“tcgetattr error”); exit ( 1 ); } if ( argv[1][0] == ‘y’ ) info.c_lflag |= ECHO; else info.c_lflag &= ~ECHO; if ( tcsetattr ( 0, TCSANOW, &info ) == -1) { perror(“tcsetattr error”); exit ( 1 ); } set the flag clear the flag
Warning When you start messing with the terminal control flags from within a program you can get yourself into a state where you don’t know what the flags are … be careful.
Consider the following program structure Molay chapter 6 do some work ask if user wants to do more get response yes no
By default the terminal is in canonical mode, so the user has the following problems: 1. The user has to press the enter key before the program can act on the input. 2. The user can type something other than ‘y’ or ‘n’ and nothing happens.
What can we do so that the user does not have to hit the Enter key to input a response?
do some work turn off canonical mode ask if user wants to do more get response turn on canonical mode yes response ? no
This works great, but we still have to reply to every illegal keystroke -- annoying! The solution: Turn off echo mode and ignore illegal keystrokes.
What if this program was running in an atm machine, and the person walked away without typing ‘y’ or ‘n’? Someone else could walk up and run a transaction using this person’s account.
Blocking vs Non-blocking I/O read The normal read operation waits until the read operation is satisfied before going on to the next instruction. do something
Blocking vs Non-blocking I/O It is possible to use fcntl or open to change The properties of a file so that when the file is read, the program does not wait, for the read to finish, but goes on immediately to the next instruction. The read instruction returns a 0 if no data has been read. read do something
Using fcntl( ) #include <sys/types.h> #include <unistd.h> #include <fcntl.h> int fcntl(int filedes, int cmd, … /* int arg */); F_DUPFD - duplicate the file descriptor F_GETFD - get the file descriptor flags F_SETFD - set the file descriptor flags F_GETFL - get the file status flags F_SETFL - set the file status flags . . .
File status flags O_RDONLY open file for reading only O_WRONLY open file for writing only O_RDWR open for reading and writing O_APPEND append on each write O_NONBLOCK non-blocking mode O_SYNC wait for writes to complete O_ASYNC asynchronous I/O (bsd and 4.3)
void set_nodelay_mode( ) { int termflags; termflags = fcntl(0, F_GETFL); termflags |= NONBLOCK; fcntl(0, F_SETFL, termflags); } get the file descriptor flags set non-blocking flag save the new flags
int get response(char* question, int maxtries) { int input; printf(“%s (y/n)?”, question); flush (stdout); while (1) { sleep(SLEEPTIME); input = tolower(get_ok_char( ) ); if ( input == ‘y’ ) return 1; if ( input == ‘n’ ) return 0; if ( maxtries -- == 0) return 2; } } print the question
int get response(char* question, int maxtries) { int input; printf(“%s (y/n)?”, question); flush (stdout); while (1) { sleep(SLEEPTIME); input = tolower(get_ok_char( ) ); if ( input == ‘y’ ) return 1; if ( input == ‘n’ ) return 0; if ( maxtries -- == 0) return 2; } } The o/s won’t flush the buffer until it sees a newline or until the program tries to read. and the read doesn’t happen until here. timeout