Форум программистов, компьютерный форум, киберфорум
C++ Builder
Войти
Регистрация
Восстановить пароль
Блоги Сообщество Поиск Заказать работу  
 
Рейтинг 4.56/9: Рейтинг темы: голосов - 9, средняя оценка - 4.56
 Аватар для KOSHAK_
114 / 3 / 0
Регистрация: 16.11.2011
Сообщений: 162

Ошибки в работе с COM портом - пакеты принимаются через один

22.10.2014, 17:24. Показов 1776. Ответов 2
Метки нет (Все метки)

Студворк — интернет-сервис помощи студентам
Доброго времени суток.
Делаю программу с использованием COM порта. Работу с COM портом осуществляю так:

C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
uint8_t input_buffer[18];
//---------------------------------------------------------------------------
 
#define BUFSIZE 18   //ёмкость буферов
 
unsigned char bufrd[BUFSIZE], bufwr[BUFSIZE];   // приёмный и передающий буферы
//---------------------------------------------------------------------------
 
HANDLE COMport;      //дескриптор порта
//структура OVERLAPPED необходима для асинхронных операций
OVERLAPPED overlapped;      //для операций чтения (поток ReadThread)
OVERLAPPED overlappedwr;    //для операций записи (поток WriteThread)
 
int handle;        //дескриптор для работы с файлом с помощью библиотеки <io.h>
String SetInitStr;
bool ConnectActive = false;
 
HANDLE reader;  //дескриптор потока чтения из порта
HANDLE writer;  //дескриптор потока записи в порт
 
DWORD WINAPI ReadThread(LPVOID);
DWORD WINAPI WriteThread(LPVOID);
 
char stop = 0, parity = 0;
 
DWORD Flag = FILE_FLAG_OVERLAPPED;
//---------------------------------------------------------------------------
 
__fastcall TForm1::TForm1(TComponent* Owner)
    : TForm(Owner)
{
}
//---------------------------------------------------------------------------
 
//главная функция потока, реализует приём байтов из COM-порта
DWORD WINAPI ReadThread(LPVOID)
{
 COMSTAT comstat;
 DWORD btr, temp, mask, signal;
 
 overlapped.hEvent = CreateEvent(NULL, true, true, NULL);
 SetCommMask(COMport, EV_RXCHAR);
 
 while(1){
    WaitCommEvent(COMport, &mask, &overlapped);
    signal = WaitForSingleObject(overlapped.hEvent, INFINITE);
 
    if(signal == WAIT_OBJECT_0){
        if(GetOverlappedResult(COMport, &overlapped, &temp, true)){
            if((mask & EV_RXCHAR)!=0){
                ClearCommError(COMport, &temp, &comstat);
 
                btr = comstat.cbInQue;
 
                if(btr){
                    ReadFile(COMport, bufrd, 18, &temp, &overlapped);
 
                    for(int n = 0; n < 18; n++){
                       input_buffer[n] = bufrd[n];
                       bufrd[n] = 0;
                    }
 
                    Form1->Timer2->Enabled = true;
                }
            }
        }
    }
 }
 
 CloseHandle(overlapped.hEvent);
}
//---------------------------------------------------------------------------
// функция записи данных в СОМ порт
DWORD WINAPI WriteThread(LPVOID)
{
 DWORD temp, signal;
 COMSTAT comstat;
 overlappedwr.hEvent = CreateEvent(NULL, true, true, NULL);
 
 while(1){
    WriteFile(COMport, bufwr, strlen(bufwr), &temp, &overlappedwr);
    signal = WaitForSingleObject(overlappedwr.hEvent, INFINITE);
 
    if((signal == WAIT_OBJECT_0) && (GetOverlappedResult(COMport, &overlappedwr, &temp, true))){
        ClearCommError(COMport, &temp, &comstat);
    }
 
    memset(bufwr, 0, BUFSIZE);
    SuspendThread(writer);
  }
}
//---------------------------------------------------------------------------
 
//функция открытия и инициализации порта
void __fastcall TForm1::COMOpen()
{
 String portname;        //имя порта (например, "COM1", "COM2" и т.д.)
 DCB dcb;                //структура для общей инициализации порта DCB
 COMMTIMEOUTS timeouts;  //структура для установки таймаутов
 
 portname = ComboBox1->Text;    //получить имя выбранного порта
 
 //открыть порт, для асинхронных операций обязательно нужно указать флаг FILE_FLAG_OVERLAPPED
 COMport = CreateFile(portname.c_str(), GENERIC_READ | GENERIC_WRITE, 0, NULL, OPEN_EXISTING, Flag, NULL);
 // - portname.c_str() - имя порта в качестве имени файла, c_str() преобразует строку типа String в строку в виде массива типа char, иначе функция не примет
 // - GENERIC_READ | GENERIC_WRITE - доступ к порту на чтение/записть
 // - 0 - порт не может быть общедоступным (shared)
 // - NULL - дескриптор порта не наследуется, используется дескриптор безопасности по умолчанию
 // - OPEN_EXISTING - порт должен открываться как уже существующий файл
 // - FILE_FLAG_OVERLAPPED - этот флаг указывает на использование асинхронных операций
 // - NULL - указатель на файл шаблона не используется при работе с портами
 
 if(COMport == INVALID_HANDLE_VALUE){
    COMClose();
    ShowMessage("Error: Can not open port");
    ConnectActive = false;
    return;
  }
 //инициализация порта
 
 dcb.DCBlength = sizeof(DCB);   //в первое поле структуры DCB необходимо занести её длину, она будет использоваться функциями настройки порта для контроля корректности структуры
 
 //считать структуру DCB из порта
 if(!GetCommState(COMport, &dcb)){  //если не удалось - закрыть порт и вывести сообщение об ошибке в строке состояния
    COMClose();
    ShowMessage("Error: Can not read DCB");
    ConnectActive = false;
    return;
 }  
 
 //инициализация структуры DCB
 dcb.BaudRate = StrToInt(ComboBox2->Text);           //задаём скорость передачи 9600 бод
 dcb.fBinary = TRUE;                                    //включаем двоичный режим обмена
 dcb.fOutxCtsFlow = FALSE;                              //выключаем режим слежения за сигналом CTS
 dcb.fOutxDsrFlow = FALSE;                              //выключаем режим слежения за сигналом DSR
 dcb.fDtrControl = DTR_CONTROL_DISABLE;                 //отключаем использование линии DTR
 dcb.fDsrSensitivity = FALSE;                           //отключаем восприимчивость драйвера к состоянию линии DSR
 dcb.fNull = FALSE;                                     //запретить приём нулевых байтов
 dcb.fRtsControl = RTS_CONTROL_DISABLE;                 //отключаем использование линии RTS
 dcb.fAbortOnError = FALSE;                             //отключаем остановку всех операций чтения/записи при ошибке
 dcb.ByteSize = StrToInt(ComboBox3->Text);       //задаём 8 бит в байте
 dcb.Parity = 0;                                   //отключаем проверку чётности
 dcb.StopBits = stop;                                   //задаём один стоп-бит
 
 //загрузить структуру DCB в порт
 if(!SetCommState(COMport, &dcb)){  //если не удалось - закрыть порт и вывести сообщение об ошибке в строке состояния
    COMClose();
    ShowMessage("Error: Can not set DCB");
    ConnectActive = false;
    return;
 }    
 
 //установить таймауты
 timeouts.ReadIntervalTimeout = 0;      //таймаут между двумя символами
 timeouts.ReadTotalTimeoutMultiplier = 0;   //общий таймаут операции чтения
 timeouts.ReadTotalTimeoutConstant = 0;         //константа для общего таймаута операции чтения
 timeouts.WriteTotalTimeoutMultiplier = 0;      //общий таймаут операции записи
 timeouts.WriteTotalTimeoutConstant = 0;        //константа для общего таймаута операции записи
 
 //записать структуру таймаутов в порт
 if(!SetCommTimeouts(COMport, &timeouts)){  //если не удалось - закрыть порт и вывести сообщение об ошибке в строке состояния
    COMClose();
    ShowMessage("Error: Can not set timeout");
    ConnectActive = false;
    return;
 }
 
 //установить размеры очередей приёма и передачи
 SetupComm(COMport, 0, 0);
 
 reader = CreateThread(NULL, 0, ReadThread, NULL, 0, NULL);         //создаём поток чтения, который сразу начнёт выполняться (предпоследний параметр = 0)
 writer = CreateThread(NULL, 0, WriteThread, NULL, CREATE_SUSPENDED, NULL); //создаём поток записи в остановленном состоянии (предпоследний параметр = CREATE_SUSPENDED)
        
 ConnectActive = true;
}
//---------------------------------------------------------------------------
 
//функция закрытия порта
void __fastcall TForm1::COMClose()
{
 if(writer){        //если поток записи работает, завершить его; проверка if(writer) обязательна, иначе возникают ошибки
    TerminateThread(writer, 0);
    CloseHandle(overlappedwr.hEvent);   //нужно закрыть объект-событие
    CloseHandle(writer);
 }
 
 if(reader){           //если поток чтения работает, завершить его; проверка if(reader) обязательна, иначе возникают ошибки
    TerminateThread(reader, 0);
    CloseHandle(overlapped.hEvent); //нужно закрыть объект-событие
    CloseHandle(reader);
 }
 
 CloseHandle(COMport);                  //закрыть порт
 COMport = 0;               //обнулить переменную для дескриптора порта
 ConnectActive = false;
}
Все работает нормально за исключением приема. Программа принимает пакеты длинной 18 байт. Сам прием идет, НО, очень интересно. Сначала принимается старый пакет, а только после этого новый, который передается в данный момент. Т.е. для того, чтобы пакет принялся, его надо отправить 2 раза. Буферы очищаю. В чем может быть проблема?
0
cpp_developer
Эксперт
20123 / 5690 / 1417
Регистрация: 09.04.2010
Сообщений: 22,546
Блог
22.10.2014, 17:24
Ответы с готовыми решениями:

Пакеты передаются, но не принимаются
1-ый комп подключен через usb кабель - все работает, 2-ой через сетевой - не пашет Модем zyxel p-660ru настроен в режим роутинг, сетевуха...

Пакеты отправляются, но не принимаются
Пытался сменить винду на Ubuntu, не пошло, старый ноут не поддерживает PAE... Снова поставил винду, и теперь пакеты отправляются, но не...

Не работает интернет, не принимаются пакеты
Перестал работать интернет, при открытии сетевого подключения отправленные пакеты идут, принятых 0, это проблема в свитче моего подъезда?...

2
504 / 247 / 75
Регистрация: 31.10.2010
Сообщений: 747
22.10.2014, 20:37
Вот рабочий код C++Builder 6, смотри, сравнивай:
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
// функции
DWORD WINAPI ReadComPort(LPVOID);   // поток опроса СОМ порта
//...
//...
//...
// поток опроса СОМ порта
DWORD WINAPI ReadComPort(LPVOID)
{
    for( ;true; ) // бесконечный цикл опроса
    {
      // ожидание события установленного в TMask
      if( !WaitCommEvent(COMport,&TMask,&Overlapp) )
      {
        if( GetLastError() == ERROR_IO_PENDING )
        {
          // ожидание приема до бесконечности
          WaitForSingleObject(Overlapp.hEvent,INFINITE);
          // получаем состояние COM в COMStat
          ClearCommError(COMport,&DWORD(NBayte),&COMStat);
          // реальное число байт в буфере
          NBayte = COMStat.cbInQue;
          // проверка на наличие в буфере необходимого числа байт
          if( NBayte >= Form1->UpDown1->Position )
          {
            // читаем данные из порта в массив COMData число байт в NBayte
            if( ReadFile(COMport,COMData,NBayte,&DWORD(NBayte),&Overlapp) );
              // посылаем сообщение
              SendMessage(Form1->Handle,WM_COMPORT,1,0);
          }
        }
      }
    }
}
Будут вопросы пиши
1
 Аватар для KOSHAK_
114 / 3 / 0
Регистрация: 16.11.2011
Сообщений: 162
24.10.2014, 11:07  [ТС]
Заработало. Спасибо
0
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
raxper
Эксперт
30234 / 6612 / 1498
Регистрация: 28.12.2010
Сообщений: 21,154
Блог
24.10.2014, 11:07
Помогаю со студенческими работами здесь

Принимаются не все пакеты UDP
Принимаю на порт посылки UDP. Длина пакетов 1400 байт, т.е. не фрагментируются. Периодичность посылок - 50 мс. Прием осуществляю по...

Windows xp. Нет интернета(пакеты не принимаются)
Здравствуйте! Помогите разобраться с проблемой. Переустановила систему,предварительно отформатировав диск. Система windows xp pro sp3 ...

Не принимаются пакеты. Сетевая карта Intel 82566DM-2
Всем привет. Такая проблема: после установки вин7 не получается подключиться к интернету. ПК HP Compaq DC 7800, сетевая Intel 82566DM-2....

Ошибки в работе датчика по HART-протоколу через COM-порт
Здравствуйте. Постараюсь как можно подробнее описать проблему. Датчик подключен к программе, через COM порт. Алгоритм следующий. Программа...


Искать еще темы с ответами

Или воспользуйтесь поиском по форуму:
3
Ответ Создать тему
Новые блоги и статьи
Настройка записи справочника по регистру сведений
Maks 29.03.2026
Решение ниже реализовано на примере нетипового справочника "ТарифыМобильнойСвязи" разработанного в конфигурации КА2, с целью учета корпоративной мобильной связи в коммерческом предприятии. . . .
Автозаполнение реквизита при выборе элемента справочника
Maks 27.03.2026
Программный код из решения ниже на примере нетипового документа "ЗаявкаНаРемонтСпецтехники" разработанного в конфигурации КА2. При выборе "Спецтехники" (Тип Справочник. Спецтехника), заполняется. . .
Сумматор с применением элементов трёх состояний.
Hrethgir 26.03.2026
Тут. https:/ / fips. ru/ EGD/ ab3c85c8-836d-4866-871b-c2f0c5d77fbc Первый документ красиво выглядит, но без схемы. Это конечно не даёт никаких плюсов автору, но тем не менее. . . всё может быть. . .
Автозаполнение реквизитов при создании документа
Maks 26.03.2026
Программный код из решения ниже размещается в модуле объекта документа, в процедуре "ПриСозданииНаСервере". Алгоритм проверки заполнения реализован для исключения перезаписи значения реквизита,. . .
Команды формы и диалоговое окно
Maks 26.03.2026
1. Команда формы "ЗаполнитьЗапчасти". Программный код из решения ниже на примере нетипового документа "ЗаявкаНаРемонтСпецтехники" разработанного в конфигурации КА2. В качестве источника данных. . .
Кому нужен AOT?
DevAlt 26.03.2026
Решил сделать простой ланчер Написал заготовку: dotnet new console --aot -o UrlHandler var items = args. Split(":"); var tag = items; var id = items; var executable = args;. . .
Отправка уведомления на почту при создании или изменении элементов справочника
Maks 24.03.2026
Программная отправка письма электронной почты на примере типового справочника "Склады" в конфигурации БП3. Перед реализацией необходимо выполнить настройку системной учетной записи электронной. . .
модель ЗдравоСохранения 5. Меньше увольнений- больше дохода!
anaschu 24.03.2026
Теперь система здравосохранения уменьшает количество увольнений. 9TO2GP2bpX4 a42b81fb172ffc12ca589c7898261ccb/ https:/ / rutube. ru/ video/ a42b81fb172ffc12ca589c7898261ccb/ Слева синяя линия -. . .
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2026, CyberForum.ru