ДЕНИС КОЛИСНИЧЕНКО
Контроль последовательных портов в Linux
Интерфейс RS-232C является одним из самых распространенных способов связи компьютеров и периферийных устройств. Кто же не помнит старый добрый Norton Commander и его функцию Связь (Link), позволяющую обмениваться файлами двум компьютерам, соединенным нуль-модемным кабелем? Интерфейс RS-232C подразумевает наличие двух типов оборудования:
- Терминального (DTE);
- Связывающего (DCE).
Терминальное устройство принимает и/или отправляет данные. К терминальному типу относятся, например, компьютеры. Терминальные устройства служат окончанием линии передачи данных, отсюда и название – terminate. Связывающее оборудование (DCE) обеспечивает передачу данных между терминальными устройствами по линии связи.
Иногда нам нужно знать состояние терминального устройства, или же мы хотим заставить связное устройство DCE работать как терминальное DTE, для этого нам нужно сделать небольшое изменение в линиях данных интерфейса RS-232C. Для этого нужно знать, какие функции выполняет тот или иной сигнал интерфейса.
Терминальное оборудование оснащено 9- или 25-контактными D-образными разъемами. На следующем рисунке изображены сами порты, а также нумерация контактов.
Рисунок 1. Последовательные порты
В таблицах 1 и 2 описаны функции контактов для 9- и 25-контактных портов соответственно.
Таблица 1. Контакты и сигналы (9 pin)
Контакт
|
Сигнал
|
Константа
|
Описание
|
1
|
CD (Carrier Detect)
|
TIOCM_CAR или TIOCM_CD
|
Определение несущей
|
2
|
RD (Receive data)
|
TIOCM_SR
|
Прием данных
|
3
|
TD (Transmit data)
|
TIOCM_ST
|
Передача данных
|
4
|
DTR (Data Terminal ready)
|
TIOCM_DTR
|
Готовность терминала
|
5
|
GND (Ground)
|
|
Земля
|
6
|
DSR (Data set ready)
|
TIOCM_DSR
|
Готовность данных (DCE)
|
7
|
RTS (Request to send)
|
TIOCM_RTS
|
Запрос передачи
|
8
|
CTS (Clear to send)
|
TIOCM_CTS
|
Готовность передачи
|
9
|
RI (RNG, Ring indicator)
|
TIOCM_RNG
|
Индикатор звонка
|
Таблица 2. Контакты и сигналы (25 pin)
Контакт
|
Сигнал
|
Описание
|
1
|
PG (Protective ground)
|
Земля экрана
|
2
|
TD (Transmit data)
|
Передача данных
|
3
|
RD (Receive data)
|
Прием данных
|
4
|
RTS (Request to send)
|
Запрос передачи
|
5
|
CTS ( Clear to send)
|
Готовность передачи
|
6
|
DSR (Data set ready)
|
Готовность данных (DCE)
|
7
|
GND (Ground)
|
Земля сигнала
|
8
|
CD (Carier Detect)
|
Определение несущей
|
9
|
Positive control Vcc
|
Положительный контроль
|
10
|
Negative control Vcc
|
Отрицательный контроль
|
11 (*)
|
QM (Quality mode)
|
Режим качества
|
12
|
SDCD (Secondary data carrier detect)
|
Определение несущей для вторичных данных (для синхронизации)
|
13
|
SCTS (Secondary data clear to send)
|
Готовность передачи вторичных данных (для синхронизации)
|
14
|
STD (Secondary transmit data)
|
Передача вторичных данных (вспомогательный канал)
|
15
|
TC (Transmit clock)
|
Синхронизация передаваемых данных
|
16
|
SRD (Secondary receive data)
|
Прием вторичных данных
|
17
|
RC (Receive clock)
|
Синхронизация принимаемых данных
|
18 (*)
|
DCR (Diffirent clock receive)
|
Сигнал синхронизации
|
19
|
SRTS (Secondary request to send)
|
Запрос передачи для вторичного канала
|
20
|
DTR (Data terminal ready)
|
Готовность терминала
|
21
|
SQ (Signal quality)
|
Качество сигнала
|
22
|
RI (RNG, Ring indicator)
|
Индикатор звонка
|
23
|
Select data speed
|
Выбор скорости передачи данных
|
24
|
TC (Terminal clock)
|
Сигнал синхронизации
|
25 (*)
|
Busy
|
Занято
|
Интерфейс RS-232C обеспечивает два независимых последовательных канала данных: основной и вспомогательный (вторичный), но на практике используется только основной, поэтому в асинхронном режиме используются только первые 9 линий из 25.
Следующая программа выводит состояние последовательного устройства.
Листинг 1. Программа для контроля последовательного порта
#include
#include
#include
#include
#include
#include
#include
int main(int argc, char **argv)
{
int fd;
int flags;
if (argc<1) {
fprintf(stderr,"Usage: pctrl serial_device");
fprintf(stderr,"For example: sns /dev/ttyS0");
exit(1);
}
if ((fd = open(argv[1], O_RDWR | O_NDELAY)) < 0) {
// Невозможно открыть порт
fprintf(stderr, "Cannot open port");
exit(1);
}
// Бесконечный цикл. Для выхода из программы нажмите Ctrl + C
while (1)
{
ioctl(fd, TIOCMGET, &flags);
/*
Используемые константы (см. таблицу 1)
#define TIOCM_LE 0x001
#define TIOCM_DTR 0x002
#define TIOCM_RTS 0x004
#define TIOCM_ST 0x008
#define TIOCM_SR 0x010
#define TIOCM_CTS 0x020
#define TIOCM_CAR 0x040
#define TIOCM_RNG 0x080
#define TIOCM_DSR 0x100
#define TIOCM_CD TIOCM_CAR
#define TIOCM_RI TIOCM_RNG
*/
// Выводим состояние устройства
if (flags & TIOCM_CD) fprintf(stderr,"1 [CD] ");
if (flags & TIOCM_SR) fprintf(stderr,"2 [RD] ");
if (flags & TIOCM_ST) fprintf(stderr,"3 [TD] ");
if (flags & TIOCM_DTR) fprintf(stderr,"4 [DTR] ");
if (flags & TIOCM_DSR) fprintf(stderr,"6 [DSR] ");
if (flags & TIOCM_RTS) fprintf(stderr,"7 [RTS] ");
if (flags & TIOCM_CTS) fprintf(stderr,"8 [CTS] ");
if (flags & TIOCM_RNG) fprintf(stderr,"9 [RNG] ");
fprintf(stderr," ");
// Засыпаем на 1 секунду
sleep(1);
}
close(fd);
}
Для компилирования программы введите команду:
gcc -o pctrl pctrl.c
Каждую секунду программа выводит на экран состояние указанного порта. Вывод программ можно увидеть на рисунке 2. Для установки какого-нибудь сигнала можно воспользоваться следующей программой:
Листинг 2. Установка сигнала RNG
#include
#include
#include
#include
#include
#include
#include
int main(int argc, char **argv)
{
int fd;
int flags=TIOCM_RNG;
if (argc<1) {
fprintf(stderr,"Usage: sns serial_device");
fprintf(stderr,"For example: sns /dev/ttyS0");
exit(1);
}
ioctl(fd, TIOCMSET, &flags);
close(fd);
}
Рисунок 2. Вывод программы
Примечание. Обе программы нужно запускать от имени пользователя root (пользователя с UID = 0). Все ваши вопросы присылайте по адресу dhsilabs@mail.ru.