Здравствуйте ХАБР. В этой, мной первый раз написанной статье, я попытаюсь рассказать про свой опыт проектирования и разработки устройства на контроллере ESP32 для мониторинга закрытой разработчиком промышленного оборудования, который является важным производственным процессом в изготовлении пластин для свинцово-кислотных аккумуляторов, что местные называют «Кюринг».
Задача была следующая – реализовать удаленный мониторинг и ведение статистики работы оборудования. Из интересующих значений выделены:
— Установленные и фактические влажность и температура в камере;
— Общее время работы и текущее время работы шага;
— Текущая программа и шаг.
На борту у оборудования сборка Siemens S7-200 Smart + HMI Proface, обмен связи MPI.
На момент реализации, нам дали информацию, что ПЛК и HMI защищенны паролем с завода, что значительно усложнило задачу и привело меня к решению сотворить внешнее устройство “снифер” на ESP32.
А может Siemens был без пароля?)
В действительности, спустя год после реализации проекта, оборудование потребовало внеочередного нашего вмешательства и мы все-таки решили попробовать подключиться к ПЛК и слить с него проект для изучения. Проблем нам это не составило, ведь никакой защиты на нем не обнаружилось. Сделай мы это раньше, устройство, речь про которое пойдет в этой статье, и не понадобилось бы, и задача закрылась бы Python + Snap7 либо NodeJS + S7Node.
И так, с чего же все началось, а началось все с определение скорости MPI. На этот момент мы уже подключились параллельно к RS485 готовым анализатором логики с использованием ПО Logic:
Получив осциллограммы сигналов, я использовал инструмент программы Analyzer и начал подбирать скорость, битность, паритет и стоп-бит, чтобы получить посылки нужного вида и без ошибок. Нужный вид посылки для себя определил, почитав статью за этом же сайте https://habr.com/ru/articles/748844/ , статья про создания своего slave устройства для Profibus, с нее же вышел на весьма подробное описание Profibus интерфейса от Макса Фелсера https://felser.ch/profibus-manual/index.html .
Таким образом, спустя непродолжительный кусок времени, параметры интерфейса были определены, и типовые посылки, из статей описанных выше, стали проглядываться:




В этой статье не будет подробно описан интерфейс, если вам это интересно, вы можете прочитать статьи, которые я упомянул выше.
Чтозж, после этого этапа, следует более интересный – найти в этом непроглядном количестве байт нас интересующие. Для этого, я уже подготовил ESP32 + MAX485 на Serial2, который имел примерно следующий вид:

Шустро написался небольшой код на Arduino, который читал ответы ПЛК на запросы HMI, и выводил мне их в исходном байтовом виде. Так я избавился от посылок запросов, токена, телеграммы без полезной нагрузки (SD1) и т.д. Для вывода сообщения я подготовил себе следующее условие:
if (SerialBuffer[0] == 0x68 and SerialBuffer[3] == 0x68 and SerialBuffer[6] == 0x08)
SerialBuffer[0] == 0x68 – Посылка с переменной длинной;
SerialBuffer[6] == 0x08 – FunctionCode ответ слейва на запрос;
В итоге у меня получились 2 вида телеграмм по длине ответа, с размером 0x27 и 0x33, так я их в итоге благополучно и начал разделять, и ложить информацию с этих телеграмм в разные массивы. Код ниже:
if (SerialBuffer[0] == 0x68 and SerialBuffer[3] == 0x68 and SerialBuffer[6] == 0x08) { int PDULenght = int(SerialBuffer[2]); switch (SerialBuffer[2]) { case 0x27: { // Serial.println("Message 0x27"); for (int i = 0; i < PDULenght; i++) { QuringInfo.Registers27[i] = SerialBuffer[i + 7]; if (i > 250) break; } TimeLastNewData = millis(); } break; case 0x33: { // Serial.println("Message 0x33"); for (int i = 0; i < PDULenght; i++) { QuringInfo.Registers33[i] = SerialBuffer[i + 7]; if (i > 250) break; } TimeLastNewData = millis(); } break; } memset(SerialBuffer, NULL, sizeof(SerialBuffer)); }
После этого, началось самое неинтересное, я добавил в код периодичный вывод в Serial состояния всех регистров в массивах QuringInfo.Registers33 и QuringInfo.Registers27, и сохранял их в Word, предварительно зафиксировав и записав все текущие значения отображаемые на HMI оборудования. Позже, томным летним вечером, я начал искать в полученных байтах записанные значения и довольно быстро нашел программу и ее шаг. С температурами и влажностями было сложнее, эти значения передавались необычно, не с помощью little-endian или big-endian, не числом с плавающей точкой – значения температур и влажностей в телеграмме передавались с умножением на 10.
То есть, когда бы на HMI была температура 44.5, то в интерфейсе она бы выглядела как два байта 0x01 и 0xBD, 0x01BD (HEX) -> 445 (DEC).
В итоге найдя таким весьма неэффективным способом все переменные, написал следующее:
QuringInfo.TempNow = ((QuringInfo.Registers27[24] << 8) | QuringInfo.Registers27[25]) / 10.0;QuringInfo.TempSet = ((QuringInfo.Registers27[26] << 8) | QuringInfo.Registers27[27]) / 10.0;QuringInfo.HumidityNow = ((QuringInfo.Registers27[30] << 8) | QuringInfo.Registers27[31]) / 10.0;QuringInfo.HumiditySet = ((QuringInfo.Registers27[28] << 8) | QuringInfo.Registers27[29]) / 10.0;QuringInfo.Program = int(QuringInfo.Registers27[33]);QuringInfo.Step = int(QuringInfo.Registers27[35]);QuringInfo.Step_Hours = int(QuringInfo.Registers33[27]);QuringInfo.Step_Minutes = int(QuringInfo.Registers33[31]);QuringInfo.Step_Seconds = int(QuringInfo.Registers33[35]);QuringInfo.Global_Hours = int(QuringInfo.Registers33[39]);QuringInfo.Global_Minutes = int(QuringInfo.Registers33[43]);QuringInfo.Global_Seconds = int(QuringInfo.Registers33[47]);
После всех перепроверок, чтобы значения соответствовали действительности в Serial мониторе, подошла очередь до проектирования печатной платы и корпуса.
Эти шаги тоже подробно описывать в этой статье не буду, боюсь что сильно раздую материал. Возможно в следующей статье подробнее опишу процесс моего проектирования и создания печатной платы, а также ее изготовления, т.к. платы перед заказом на Китае, мы изготавливаем самостоятельно на фрезерном ЧПУ CNC3018 и недавно появившемся лазерном ЧПУ.
Примерно так устройство встало в шкаф (почти как там и была):

На момент полной реализации и отладки устройства, мой коллега уже полностью подготовил сервер, написанный на nodejs, обмен связи 6 устройств с сервером происходит по WebSocket и спустя примерно месяц после первого подключения к RS485 имеем такой результат:



Поставленная задача полностью выполнена, за исключением того, что перехват этих данным сотворенным устройством возможен, только когда HMI на корневой странице с данными, когда мы перейдем в настройки или просто в меню, данным на интерфейсе уже не будет. Это решилось настройкой из-под встроенного меню настройки HMI Proface, где мы установили Standby mode по истечению 1 минуты бездействия, который вернет нас на нужную страницу, на случай, если оператор забудет это сделать)
ссылка на оригинал статьи https://habr.com/ru/articles/1040304/