/*! Copyright 2017 АО "НИИЭТ" и ООО "НПФ ВЕКТОР" Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. \file V_ModBus.c \brief Драйвер ModBus (см. TModBus) 24-03-2017 Перенос драйвера на К1921ВК01Т НИИЭТ. Сильный рефакторинг. 13-08-2012 Теперь поле type структуры MB_Record используется как признак записи/чтения для не битовых HR/IR-переменных (раньше там был только признак записи). 24-01-2012 Теперь проще выбирать между SciaRegs, ScibRegs и ScicRegs. Это сделано дефайном. Переименована структура RS в MODBUS_RS 26-02-2011 Исправлены какие-то баги, добавлена возможность отслеживать запись в объекты типа "регистр" за счет изменения номера бита в словаре, поддержка большего кол-ва процессоров. 27-11-2009 Сделана доработка драйвера для поддержки broadcast (широковещательных) сообщений. Естесственно, что поддерживаются только команды записи (0x5 - write coils и 0x6 - write single register), т.к. broadcast сообщения являются неподтверждаемыми сообщениями. Иными словами читать бесполезно, т.к. ответа не придет... (Алямкин) 21-10-2008 Исправлена ошибка переполнения входного регистра - при переполнении выставлялась frame error (из драйвера SYS_VID800) 29-01-2008 Иправлена ошибка расчета таймаутов 28-01-2008 Иправлена ошибка в функции пойска нужного элемента словаря 29-11-2007 Драйвер переделан для работы с МК17.1 1.Исправлена ошибка при обнулении битовых регистров 2.Исправлен неправильный перебор элементов в словаре 3.Добавлена возможность автоматического перезапуска драйвера при большом количестве ошибок на линии Замечание. Особенность реализации драйвера не предполагает наличия регистра № 0х0000 в словаре Для устранения недостатка необходимо в инициализации переделать пойск последних элементов массивов с поля номера регистра на поле адреса переменной регистра. 16-11-2006 Release Rev 1.0 \author Alecksey Anuchin \version v 1.2 24/03/2017 */ /** \addtogroup V_ModBus */ /*@{*/ #include "V_ModBus.h" #include "main.h" //Макросы для переключения RS485 между приемом и передачей #define MODBUS_RX_RS485 GPIOM->DATAOUTCLR = (1 << 2) //M2 #define MODBUS_TX_RS485 GPIOM->DATAOUTSET = (1 << 2) //M2 #define MODBUS_GPIO_VALUE (GPIOK->DATA>>10) & 3 //Макрос для осциллографирования ножек UART (удобно для отладки, когда вообще ничего не работает) //Частота тактирования блока UART (настраивается в инициализации микроконтроллера) #define UART_CLOCK_FREQ 25000000 void ModBus_RS_Init(TModBus *p) { p->MBInternals.NT_UART = UART2; //Используемый в драйвере номер UART (1,2,3) p->MBInternals.NT_UART->CR_bit.UARTEN = 1; // Разрешить работу UART //Текущая реализация драйвера не использует FIFO, обрабатывает данные по одному байту //и вызывается в прерывании 10кГц. Поэтому частоты обмена больше 57600 не поддерживаются (будет терять байты) if (p->BaudRate > 57600) p->BaudRate = 57600; // Настройка частоты в двух регистрах: // задание делителя частоты для получения BaudRate // Т.к. блок UART тактируется с частотой 25 МГц (так настроено в ините микроконтроллера), // то для получения частоты, например, 57600 бит/с необходим // делитель 25000000 / (16 * 57600) = 27,126736111111111111111111111111 // Целая часть I = 27 // Дробная часть F = (int)( 0.126736111*64 + 0.5 ) = 8 // Формулы см. в документации float DivResFloat = (float) UART_CLOCK_FREQ / (16 * p->BaudRate); long DivResInt = (long) DivResFloat; long DivResFrac = (int) ((DivResFloat - DivResInt) * 64 + 0.5); p->MBInternals.NT_UART->IBRD_bit.DIVINT = DivResInt; p->MBInternals.NT_UART->FBRD_bit.DIVFRAC = DivResFrac; // 8 бит данных, 1 стоп бит, без контроля четности p->MBInternals.NT_UART->LCRH_bit.SPS = 0; // Нет проверки четности p->MBInternals.NT_UART->LCRH_bit.WLEN = 3; // Длина посылки 8 бит p->MBInternals.NT_UART->LCRH_bit.FEN = 0; // Не использовать FIFO p->MBInternals.NT_UART->CR_bit.TXE = 1; // Разрешить приём p->MBInternals.NT_UART->CR_bit.RXE = 1; // Разрешить передачу p->MBInternals.NT_UART->LCRH_bit.STP2 = 0; // 1 стоп-бит } void ModBus_ResetError(TModBus *p) { // Биты 0-3 - биты ошибок, сбрасываются записью в них "1" p->MBInternals.NT_UART->RSR |= 0x000F; } int16 ModBus_CheckDataReady(TModBus *p) { // return 1 - если буфер приёмника не пуст return (!p->MBInternals.NT_UART->FR_bit.RXFE); } int16 ModBus_CheckTxEmpty(TModBus *p) //Если все отправлено (пуст и основной и теневой регистры) { // return 1 - если буфер передатчика пуст return (p->MBInternals.NT_UART->FR_bit.TXFE); } int16 ModBus_CheckTxReady(TModBus *p) //В передатчик можно положить следующий символ (буфер свободен) { // "1" - если буфер передатчика пуст return (p->MBInternals.NT_UART->FR_bit.TXFE); } void ModBus_Tx(TModBus *p, int16 data) // Передача данных { Uint8 tmp_data = data; p->MBInternals.NT_UART->DR_bit.DATA = tmp_data; } //!Инициализация. //!настройка таймаутов, //!обработка словаря объектов для последующего быстрого поиска //! \memberof TModBus void ModBus_Init(TModBus *p) { Uint16 x; ModBus_RS_Init(p); p->MBInternals.TimeOut1_5 = (int16) (((int32) p->ExecutionFreq * (int32) 17) / p->BaudRate + 2); // 17 ~ 11*1.5 p->MBInternals.TimeOut2_5 = (int16) (((int32) p->ExecutionFreq * (int32) 28) / p->BaudRate); // 28 ~ 11*2.5 p->MBInternals.RecievedCounter = -1; ////// HR // поиск размера массива HR x = 0; while (mbodHR[x].index != 0) x++; p->MBInternals.NumOfHRs = x; // поиск среднего элемента массива HR p->MBInternals.MiddleOfHRs = x; x = 0; while ((p->MBInternals.MiddleOfHRs >> 1) != 0) { p->MBInternals.MiddleOfHRs >>= 1; x = x + x + 1; } p->MBInternals.MiddleOfHRs = x; // поиск начального инкремента для метода последовательного // приближения p->MBInternals.InitialStepOfHRs = 1; while (x != 0) { x >>= 1; p->MBInternals.InitialStepOfHRs <<= 1; } p->MBInternals.InitialStepOfHRs >>= 1; ////// Coils // поиск размера массива C x = 0; while (mbodC[x].index != 0) x++; p->MBInternals.NumOfCs = x; // поиск среднего элемента массива C p->MBInternals.MiddleOfCs = x; x = 0; while ((p->MBInternals.MiddleOfCs >> 1) != 0) { p->MBInternals.MiddleOfCs >>= 1; x = x + x + 1; } p->MBInternals.MiddleOfCs = x; // поиск начального инкремента для метода последовательного // приближения p->MBInternals.InitialStepOfCs = 1; while (x != 0) { x >>= 1; p->MBInternals.InitialStepOfCs <<= 1; } p->MBInternals.InitialStepOfCs >>= 1; // Discrete Inputs // поиск размера массива DI x = 0; while (mbodDI[x].index != 0) x++; p->MBInternals.NumOfDIs = x; // поиск среднего элемента массива DI p->MBInternals.MiddleOfDIs = x; x = 0; while ((p->MBInternals.MiddleOfDIs >> 1) != 0) { p->MBInternals.MiddleOfDIs >>= 1; x = x + x + 1; } p->MBInternals.MiddleOfDIs = x; // поиск начального инкремента для метода последовательного // приближения p->MBInternals.InitialStepOfDIs = 1; while (x != 0) { x >>= 1; p->MBInternals.InitialStepOfDIs <<= 1; } p->MBInternals.InitialStepOfDIs >>= 1; ////// IR // поиск размера массива HR x = 0; while (mbodIR[x].index != 0) x++; p->MBInternals.NumOfIRs = x; // поиск среднего элемента массива HR p->MBInternals.MiddleOfIRs = x; x = 0; while ((p->MBInternals.MiddleOfIRs >> 1) != 0) { p->MBInternals.MiddleOfIRs >>= 1; x = x + x + 1; } p->MBInternals.MiddleOfIRs = x; // поиск начального инкремента для метода последовательного // приближения p->MBInternals.InitialStepOfIRs = 1; while (x != 0) { x >>= 1; p->MBInternals.InitialStepOfIRs <<= 1; } p->MBInternals.InitialStepOfIRs >>= 1; for (x = 0; x < 1000; x++) ; } //!Расчет //!Фунция обработки посылок ModBus //! \memberof TModBus void ModBus_Execute(TModBus *p) { Uint16 x; volatile Uint16 RxReg; if (ModBus.Enabled == 1) { // Если включен MODBUS p->OfflineCounter += 1; //долго нет собщений по сети? if (p->OfflineCounter > p->OfflineCounterMax) { p->isOnline = 0; //Мы не в сети p->OfflineCounter = p->OfflineCounterMax + 1; //чтоб не убежал } else p->isOnline = 1; //в сети //Контроль ошибок if (p->errorCode) { //Если висит ошибка p->error += 1; if (p->AutoRestart == 1) //Если разрешен авто сброс ошибок ModBus_ResetError(p); } if (p->clear_error == 1) //Если выставлен флаг однократного ручного сброса ошибки { ModBus_ResetError(p); p->clear_error = 0; p->error = 0; } //Текущее значение GPIO для отладки p->GPIOsValue = MODBUS_GPIO_VALUE; if (p->MBInternals.TxState < 1) //TxState==0 - передача не идет if (ModBus_CheckDataReady(p)) //Если есть готовые принятные данные (байт) { p->MBInternals.TimeOut = p->MBInternals.TimeOut1_5; //Таймаут равный полтора символа RxReg = p->MBInternals.NT_UART->DR; //Прием очередного символа p->errorCode = (RxReg & 0x0F00) >> 8; RxReg = RxReg & 0xFF; p->received_data = RxReg; //Перекладывание принятых данных для отладки p->ReceiveByteCounter++; if (p->MBInternals.InCount == MODBUS_MAX_RX_COUNT) //Принятно слишком много байт return; p->MBInternals.In[p->MBInternals.InCount++] = RxReg; //Принятые данные в массив p->MBInternals.RxCRC ^= RxReg; //Сразу же расчет контрольной суммы for (RxReg = 0; RxReg < 8; RxReg++) //Это тоже продолжение расчета контрольной суммы if (p->MBInternals.RxCRC & 0x01) { p->MBInternals.RxCRC >>= 1; p->MBInternals.RxCRC ^= p->MBInternals.MagicNumber; } else p->MBInternals.RxCRC >>= 1; //больше ничего, если есть принятые данные, не делается return; } if (p->MBInternals.TimeOut == 0) //Когда таймаут вышел, его обрабатывает условие p->MBInternals.TimeOut==1, а равен нулю он становится после приема { p->MBInternals.InCount = 0; //Обнуление счетчика массива p->MBInternals.RxCRC = 0xFFFF; //Инициализация контрольной суммы на прием if (ModBus_CheckTxEmpty(p)) //Если буфер на отправку пуст { MODBUS_RX_RS485; //Переключение RS485 на прием p->MBInternals.TxState = 0; //TxState==0 - передача не идет p->MBInternals.TxCRC = 0xFFFF; //Инициализация контрольной суммы на передачу } return; } if (p->MBInternals.TimeOut == 1) //; Сюда попадаем, когда межу символами появляется пауза в 1,5 символа if (p->MBInternals.InCount != 0) //; Если количество входных данных не равно нулю, то рассматривем, что пришло { //поддержка broadcast осуществляется тут (по другому я незнаю как еще - Алямкин) //далее переход в передачу - нам туда не нужно, потому обрабатываем здесь //обрабатывать имеет смысл только команды на запись (0x5 - write coils и 0x6 - write single register). if (p->MBInternals.In[0] == 0)//Входной массив, байт адреса равен 0 - широковещательный { //0x5 - write coils if (p->MBInternals.In[1] == 5) { x = (p->MBInternals.In[4] << 8) + p->MBInternals.In[5]; //Данные //0 - если нашла, 3 - если запрос некорректен, 2 - ошибка в поиске p->MBInternals.temp = ModBus_FindVocAddr(p, mbodC, p->MBInternals.NumOfCs, 1, (p->MBInternals.In[2] << 8) + p->MBInternals.In[3], 1, p->MBInternals.MiddleOfCs, p->MBInternals.InitialStepOfCs); if ((x != 0xFF00) && (x != 0x0000)) //Данные не валидны, для битовой переменной должны быть FF00 или 0000 p->MBInternals.temp = 3;//делается как будто ModBus_FindVocAddr вернула некорректный запрос if (p->MBInternals.temp == 0)//если ModBus_FindVocAddr нашла адрес { if (p->MBInternals.In[4] == 0) // reset bit *(Uint16 *) (p->MBInternals.Addr->addr) &= (0xFFFF - (1 << p->MBInternals.Addr->type_or_acc)); else // set bit *(Uint16 *) (p->MBInternals.Addr->addr) |= (1 << p->MBInternals.Addr->type_or_acc); } } if (p->MBInternals.In[1] == 6) //0x6 - write HR { p->MBInternals.temp = ModBus_FindVocAddr(p, mbodHR, p->MBInternals.NumOfHRs, 16, (p->MBInternals.In[2] << 8) + p->MBInternals.In[3], 1, p->MBInternals.MiddleOfHRs, p->MBInternals.InitialStepOfHRs); if (p->MBInternals.temp == 0) //если ModBus_FindVocAddr нашла адрес { *(Uint16 *) (p->MBInternals.Addr->addr) = (p->MBInternals.In[4] << 8) + p->MBInternals.In[5]; } } //делаем то же самое, что и ниже для правильной отбраковки сообщения и выходим p->MBInternals.InCount = 0; //обнуление массива принятых байт p->MBInternals.TimeOut = 0; //обнуление таймаута p->MBInternals.RxCRC = 0xFFFF; //инициализация контрольной суммы return; } if (p->MBInternals.In[0] != p->RSNodeID) //; Если номер узла не совпал с нашим, то выкидываем посылку. { p->MBInternals.InCount = 0; p->MBInternals.TimeOut = 0; p->MBInternals.RxCRC = 0xFFFF; return; } if (p->MBInternals.RxCRC != 0) //; Если не сошлась контрольная сумма, то выкидываем посылку. { p->MBInternals.InCount = 0; p->MBInternals.TimeOut = 0; p->MBInternals.RxCRC = 0xFFFF; return; } if (p->MBInternals.RecievedCounter < 0) //!!! //; Если еще не проинициализирован счетчик между фреймов (-1), то { //; инициализируем счетчик и выходим. p->MBInternals.RecievedCounter = p->MBInternals.TimeOut2_5; return; } if (p->MBInternals.RecievedCounter > 0) //!!! //; Если счетчик паузы между фреймов еще положительный, то уменьшаем его { //; и выходим p->MBInternals.RecievedCounter--; return; } //; Если счетчик RecievedCounter досчитал до нуля, то можно переключаться на передачу. p->MBInternals.RecievedCounter = -1; p->ReceivePacketCounter++; MODBUS_TX_RS485; //RS485 на передачу p->MBInternals.TxState = 1; //Передача p->MBInternals.RxCRC = 0xFFFF; p->MBInternals.TimeOut = p->MBInternals.TimeOut2_5; ModBus_Tx(p, p->MBInternals.In[0]); //Передается первый байт - адрес устройства, почему-то взят из приема... p->MBInternals.TxCRC ^= p->MBInternals.In[0]; //сразу расчет контрольноу суммы на передачу //и это тоже расчет контрольной суммы for (RxReg = 0; RxReg < 8; RxReg++) if (p->MBInternals.TxCRC & 0x01) { p->MBInternals.TxCRC >>= 1; p->MBInternals.TxCRC ^= p->MBInternals.MagicNumber; } else p->MBInternals.TxCRC >>= 1; return; } else //количество принятых данных равно нулю.. почему-то { p->MBInternals.TimeOut = 0; return; } p->MBInternals.TimeOut--; //отсчитывается таймаут if (p->MBInternals.TxState < 1) //если TxState==0 - нет передачи, т выход, т.к. ниже - передача. return; if (ModBus_CheckTxReady(p) == 0) //если передатчик занят return; //Инициализация таймаута p->MBInternals.TimeOut = p->MBInternals.TimeOut2_5; // с этого момента включается дискретный автомат // протокола //p->TxState=0; if (p->MBInternals.TxState == 1) //Состояние передачи 1 { p->OfflineCounter = 0; //Счетчик для контроля пересылаемых пакетов ModBus p->MBInternals.temp = p->MBInternals.In[1]; //Принятый массив, тут номер функции if (p->MBInternals.temp == 1) // Функция read Coils { p->MBInternals.ToSend = p->MBInternals.temp; //для отправки - номер функции p->MBInternals.temp = ModBus_FindVocAddr(p, mbodC, p->MBInternals.NumOfCs, 1, (p->MBInternals.In[2] << 8) + p->MBInternals.In[3], (p->MBInternals.In[4] << 8) + p->MBInternals.In[5], p->MBInternals.MiddleOfCs, p->MBInternals.InitialStepOfCs); if (p->MBInternals.temp == 0) //в словаре найден элемент { p->MBInternals.TxState = 10; } else //ошибка, в словаре элемент не найден { p->MBInternals.ToSend |= 0x80; //добавляется маска ошибки p->MBInternals.TxState = p->MBInternals.temp + 3; } } else if (p->MBInternals.temp == 2) // Код функции read DI { p->MBInternals.ToSend = p->MBInternals.temp; //для отправки - номер функции p->MBInternals.temp = ModBus_FindVocAddr(p, mbodDI, p->MBInternals.NumOfDIs, 1, (p->MBInternals.In[2] << 8) + p->MBInternals.In[3], (p->MBInternals.In[4] << 8) + p->MBInternals.In[5], p->MBInternals.MiddleOfDIs, p->MBInternals.InitialStepOfDIs); if (p->MBInternals.temp == 0) //в словаре найден элемент { p->MBInternals.TxState = 10; } else //ошибка, в словаре элемент не найден { p->MBInternals.ToSend |= 0x80; //добавляется маска ошибки p->MBInternals.TxState = p->MBInternals.temp + 3; } } else if (p->MBInternals.temp == 3) // Функция read HR { p->MBInternals.ToSend = p->MBInternals.temp; p->MBInternals.temp = ModBus_FindVocAddr(p, mbodHR, p->MBInternals.NumOfHRs, 16, (p->MBInternals.In[2] << 8) + p->MBInternals.In[3], (p->MBInternals.In[4] << 8) + p->MBInternals.In[5], p->MBInternals.MiddleOfHRs, p->MBInternals.InitialStepOfHRs); if (p->MBInternals.temp == 0) //в словаре найден элемент { p->MBInternals.TxState = 7; } else //ошибка, в словаре элемент не найден { p->MBInternals.ToSend |= 0x80; //добавляется маска ошибки p->MBInternals.TxState = p->MBInternals.temp + 3; } } else if (p->MBInternals.temp == 4) // Функция read IR { p->MBInternals.ToSend = p->MBInternals.temp; p->MBInternals.temp = ModBus_FindVocAddr(p, mbodIR, p->MBInternals.NumOfIRs, 16, (p->MBInternals.In[2] << 8) + p->MBInternals.In[3], (p->MBInternals.In[4] << 8) + p->MBInternals.In[5], p->MBInternals.MiddleOfIRs, p->MBInternals.InitialStepOfIRs); if (p->MBInternals.temp == 0) //в словаре найден элемент { p->MBInternals.TxState = 7; } else //ошибка, в словаре элемент не найден { p->MBInternals.ToSend |= 0x80; //добавляется маска ошибки p->MBInternals.TxState = p->MBInternals.temp + 3; } } else if (p->MBInternals.temp == 5) // Функция write Coil { p->MBInternals.ToSend = p->MBInternals.temp; x = (p->MBInternals.In[4] << 8) + p->MBInternals.In[5]; //Данные p->MBInternals.temp = ModBus_FindVocAddr(p, mbodC, p->MBInternals.NumOfCs, 1, (p->MBInternals.In[2] << 8) + p->MBInternals.In[3], 1, p->MBInternals.MiddleOfCs, p->MBInternals.InitialStepOfCs); if ((x != 0xFF00) && (x != 0x0000)) //Данные не валидны - можно только 0xFF00 и 0x0000 p->MBInternals.temp = 3; //Как будто бы ModBus_FindVocAddr вернула ошибку if (p->MBInternals.temp == 0) //в словаре найден элемент { p->MBInternals.TxState = 12; if (p->MBInternals.In[4] == 0) // Операция над словарем - reset bit *(Uint16 *) (p->MBInternals.Addr->addr) &= (0xFFFF - (1 << p->MBInternals.Addr->type_or_acc)); else // set bit *(Uint16 *) (p->MBInternals.Addr->addr) |= (1 << p->MBInternals.Addr->type_or_acc); p->Refresh = 1; } else //ошибка, в словаре элемент не найден { p->MBInternals.ToSend |= 0x80; //добавляется маска ошибки p->MBInternals.TxState = p->MBInternals.temp + 3; } } else if (p->MBInternals.temp == 6) // Функция write HR { p->MBInternals.ToSend = p->MBInternals.temp; p->MBInternals.temp = ModBus_FindVocAddr(p, mbodHR, p->MBInternals.NumOfHRs, 16, (p->MBInternals.In[2] << 8) + p->MBInternals.In[3], 1, p->MBInternals.MiddleOfHRs, p->MBInternals.InitialStepOfHRs); if (p->MBInternals.temp == 0) //в словаре найден элемент { p->MBInternals.TxState = 12; //Операция над словарем - запись регистра *(Uint16 *) (p->MBInternals.Addr->addr) = (p->MBInternals.In[4] << 8) + p->MBInternals.In[5]; p->MBInternals.Addr->type_or_acc = MB_VAR_WRITE; } else //ошибка, в словаре элемент не найден { p->MBInternals.ToSend |= 0x80; //добавляется маска ошибки p->MBInternals.TxState = p->MBInternals.temp + 3; } } else //никакая из поддерживаемых функций { p->MBInternals.ToSend = p->MBInternals.temp | 0x80; //добавляется маска ошибки p->MBInternals.TxState = 2; } } else if (p->MBInternals.TxState == 12) // send addr hi for write C { p->MBInternals.TxState = 13; p->MBInternals.ToSend = p->MBInternals.In[2]; } else if (p->MBInternals.TxState == 13) // send addr low for write C { p->MBInternals.TxState = 14; p->MBInternals.ToSend = p->MBInternals.In[3]; } else if (p->MBInternals.TxState == 14) // send data hi for write C { p->MBInternals.TxState = 15; p->MBInternals.ToSend = p->MBInternals.In[4]; } else if (p->MBInternals.TxState == 15) // send data hi for write C { p->MBInternals.TxState = 3; p->MBInternals.ToSend = p->MBInternals.In[5]; } else if (p->MBInternals.TxState == 10) // send length read Coils { p->MBInternals.TxState = 11; p->MBInternals.In[5] = p->MBInternals.In[5] + (p->MBInternals.In[4] << 8); p->MBInternals.ToSend = (p->MBInternals.In[5] + 7) >> 3; } else if (p->MBInternals.TxState == 11) // send data read Coils { p->MBInternals.ToSend = 0; for (x = 0; x < 8; x++) //Можно упаковать 8 битовых переменных за раз { if (p->MBInternals.In[5] != 0) //количество запрашиваемых элементов словаря { //Отправка элемента словаря p->MBInternals.ToSend |= (((*(Uint16 *) (p->MBInternals.Addr->addr)) >> (p->MBInternals.Addr->type_or_acc)) & 0x01) << x; p->MBInternals.Addr++; p->MBInternals.In[5]--; } if (p->MBInternals.In[5] == 0) //Если все отправили p->MBInternals.TxState = 3; } } else if (p->MBInternals.TxState == 7) // send length read HR { p->MBInternals.TxState = 8; p->MBInternals.ToSend = p->MBInternals.In[5] << 1; } else if (p->MBInternals.TxState == 8) // send HR high word { p->MBInternals.TxState = 9; //x= *p->MBInternals.Addr->addr; p->MBInternals.temp = *(Uint16 *) (p->MBInternals.Addr->addr); p->MBInternals.ToSend = p->MBInternals.temp >> 8; } else if (p->MBInternals.TxState == 9) // send HR low word { p->MBInternals.ToSend = p->MBInternals.temp & 0xFF; p->MBInternals.Addr->type_or_acc = MB_VAR_READ; p->MBInternals.Addr += 1; p->MBInternals.In[5]--; //уменьшение кол-ва запрошенных переменных во входном массиве if (p->MBInternals.In[5] == 0) //Если это количество равно нулю (все переменные отправили) p->MBInternals.TxState = 3; else p->MBInternals.TxState = 8; } else if (p->MBInternals.TxState == 2) // no such function { p->MBInternals.ToSend = 1; p->MBInternals.TxState = 3; } else if (p->MBInternals.TxState == 5) // send error 2 { p->MBInternals.ToSend = 2; p->MBInternals.TxState = 3; } else if (p->MBInternals.TxState == 6) // send error 3 { p->MBInternals.ToSend = 3; p->MBInternals.TxState = 3; } else if (p->MBInternals.TxState == 3) // tx crc low { p->MBInternals.ToSend = p->MBInternals.TxCRC & 0x00FF; p->MBInternals.TxState = 4; } else if (p->MBInternals.TxState == 4) // tx crc high { p->MBInternals.ToSend = p->MBInternals.TxCRC & 0x00FF; p->MBInternals.TxState = 0; } else p->MBInternals.TxState = 0; p->MBInternals.TimeOut = p->MBInternals.TimeOut2_5; ModBus_Tx(p, p->MBInternals.ToSend); p->MBInternals.TxCRC ^= p->MBInternals.ToSend; for (RxReg = 0; RxReg < 8; RxReg++) if (p->MBInternals.TxCRC & 0x01) { p->MBInternals.TxCRC >>= 1; p->MBInternals.TxCRC ^= p->MBInternals.MagicNumber; } else p->MBInternals.TxCRC >>= 1; return; } } //!Вспомогательная функция поиска //!функция ищет элемент в таблице и возвращает ссылку на него в p->MBInternals.Addr //!возвращает: //! 0 - если нашла //! 3 - если запрос некорректен //! 2 - ошибка в пойске //! \memberof TModBus int16 ModBus_FindVocAddr(TModBus *p, volatile MB_Record /*const*/Table[], Uint16 TableSize, int16 Type, Uint16 Index, int16 NumOfIndexs, int16 Mid, int16 Step) { int16 x = 0; if (Type == 16) { if (NumOfIndexs == 0) return (3); if (NumOfIndexs > 0x7D) return (3); } else { if (NumOfIndexs == 0) return (3); if (NumOfIndexs > 0x7D0) return (3); } x += Mid; Step <<= 1; do { Step >>= 1; if (Index == Table[x].index) { if ((x + NumOfIndexs) > TableSize) return (2); p->MBInternals.Addr = &Table[x]; return (0); } else if (Index < Table[x].index) x -= Step; else { x += Step; if (x >= TableSize) x = (TableSize - 1); } } while (Step != 0); return (2);; } /*@}*/