Основы программирования в Linux - Страница 87

Изменить размер шрифта:

$ stty -echo

Примечание

Не забудьте применить команду

stty echo
для возврата отображения после ваших экспериментов!

Скорость терминала

Последняя функция, обслуживаемая структурой

termios
, — манипулирование скоростью линии передачи. Для этой скорости не определен отдельный элемент структуры; она управляется вызовами функций. Скорости ввода и вывода обрабатываются отдельно.

Далее приведены четыре прототипа вызовов:

#include <termios.h> 

speed_t cfgetispeed(const struct termios *);

speed_t cfgetospeed(const struct termios *);

int cfsetispeed(struct termios *, speed_t speed);

int cfsetospeed(struct termios *, speed_t speed);

Обратите внимание на то, что они воздействуют на структуру

termios
, а не непосредственно на порт. Это означает, что для установки новой скорости вы должны считать текущие установки с помощью функции
tcgetattr
, задать скорость, применив приведенные вызовы, и затем записать структуру
termios
обратно с помощью функции
tcsetattr
. Скорость линии передачи изменится только после вызова
tcsetattr
.

В вызовах перечисленных функций допускается задание разных значений скорости

speed
, но к основным относятся следующие константы:

□ 

B0
— отключение терминала;

□ 

B1200
— 1200 бод;

□ 

B2400
— 2400 бод;

□ 

B9600
— 9600 бод;

□ 

B19200
— 19 200 бод;

□ 

B38400
— 38 400 бод.

Не существует скоростей выше 38 400 бод, задаваемых стандартом, и стандартного метода обслуживания последовательных портов на более высоких скоростях.

Примечание

В некоторых системах, включая Linux, для выбора более высоких скоростей определены константы

В57600
,
B115200
и
В230400
. Если вы пользуетесь более старой версией ОС Linux и эти константы недоступны, можно применить команду
setserial
для получения нестандартных скоростей 57 600 и 115 200. В этом случае указанные скорости будут использоваться при выборе константы B38400. Оба эти метода непереносимы, поэтому применяйте их с осторожностью.

Дополнительные функции

Есть небольшое число дополнительных функций для управления терминалами. Они работают непосредственно с дескрипторами файлов без необходимости считывания и записывания структур типа

termios
.

#include <termios.h>

int tcdrain(int fd);

int tcflow(int fd, int flowtype);

int tcflush(int fd, int in_out_selector);

Функции предназначены для следующих целей:

□ 

tcdrain
— заставляет вызвавшую программу ждать до тех пор, пока не будет отправлен весь поставленный в очередь вывод;

□ 

tcflow
— применяется для приостановки или возобновления вывода;

□ 

tcflush
— может применяться для отказа от входных или выходных данных либо и тех, и других.

Теперь, когда мы уделили довольно много внимания структуре

termios
, давайте рассмотрим несколько практических примеров. Возможно, самый простой из них — отключение отображения при чтении пароля (упражнение 5.4). Это делается сбрасыванием флага
echo
.

Упражнение 5.4. Программа ввода пароля с применение
termios

1. Начните вашу программу password.с со следующих определений:

#include <termios.h>

#include <stdio.h>

#include <stdlib.h>

#define PASSWORD_LEN 8

int main() {

 struct termios initialrsettings, newrsettings;

 char password[PASSWORD_LEN + 1];

2. Далее добавьте строку, считывающую текущие установки из стандартного ввода и копирующую их в только что созданную вами структуру типа

termios
:

 tcgetattr(fileno(stdin), &initialrsettings);

3. Создайте копию исходных установок, чтобы восстановить их в конце. Сбросьте флаг

ECHO
в переменной
newrsettings
и запросите у пользователя его пароль:

 newrsettings = initialrsettings;

 newrsettings.с_lflag &= ~ЕСНО;

 printf("Enter password: ");

4. Далее установите атрибуты терминала в newrsettings и считайте пароль. И наконец, восстановите первоначальные значения атрибутов терминала и выведите пароль на экран, чтобы свести на нет все предыдущие усилия по обеспечению безопасности:

 if (tcsetattr(fileno(stdin), TCSAFLUSH, &newrsettings) != 0) {

  fprintf(stderr, "Could not set attributesn");

 } else {

  fgets(password, PASSWORD_LEN, stdin);

  tcsetattr(fileno(stdin), TCSANOW, &initialrsettings);

  fprintf(stdout, "nYou entered %sn", password);

 }

 exit(0);

}

Когда вы выполните программу, то увидите следующее:

$ ./password

Enter password: You entered hello

$
 

Как это работает

В этом примере слово

hello
набирается на клавиатуре, но не отображается на экране в строке приглашения
Enter password:
. Никакого вывода нет до тех пор, пока пользователь не нажмет клавишу .

Будьте осторожны и изменяйте с помощью конструкции

X&=~FLAG
(которая очищает бит, определенный флагом
FLAG
в переменной
X
) только те флаги, которые вам нужно изменить. При необходимости можно воспользоваться конструкцией
X|=FLAG
для установки одиночного бита, определяемого
FLAG
, хотя в предыдущем примере она не понадобилась.

Оригинальный текст книги читать онлайн бесплатно в онлайн-библиотеке Knigger.com