вторник, 14 июля 2020 г.

Дата и время: S7-1200 и панели оператора Siemens

Постоянно приходится иметь дело с настройкой даты/времени панелей оператора Siemens и контроллеров S7-1200. В этой статье я расскажу как делается синхронизация даты/времени панелей операторов (HMI) с этим PLC. Основное предназначение такого мероприятия: правильное отображение даты/времени в аварийных и иных сообщениях понели оператора, и чтобы эти дату и время не нужно было задавать каждый раз при включении панели.

Принцип


Принцип установки и синхронизации даты/времени выглядит так:

- дата и время хранятся на PLC, поскольку в этом устройстве они не сбиваются при отключении питания

- HMI периодически синхронизирует свои дату/время с PLC

- на HMI есть ячейки, которые позволяют изменять дату и время PLC и, таким образом, менять дату и время HMI (которая синхронизирована с PLC)

Возможности


Возможности предоставляются следующие.

Со стороны HMI:

1. Считывать текущее дату/время HMI и использовать её для каких-то целей на ПЛК. Для этого надо выделить ячейки памяти в Data Block'е PLC для сохранения этих даты/времени и задать ячейку памяти Job MailBox - это та ячейка, которая воспринимается панелью оператора как внешняя команда, в частности, значение 40 в этой ячейке - это команда скопировать время HMI в заданную область память PLC.

2. Дать задание HMI обновлять один раз в заданный период времени (раз в секунду, минуту, час...) свои дату/время в соответствии со значениями, задаваемыми в заранее выделенной для этого области PLC. Т.е. панель оператора сама не запрашивает у PLC текущие дату/время, но если алгоритм работы PLC будет пересылать свои значения в определённые ячейки, то панель с ними синхронизируется.

Со стороны PLC:

1. Считывать текущие дату/время PLC и сохранять их в виде значений в Data Block'ах.

2. Задавать дату и время, хранящуюся в Data Block'ах, в качестве даты/времени PLC.



Отличия WinCC Flexible 2008 / WinCC Flexible Smart от WinCC TIA Portal

Стóит заметить, что есть разница между WinCC, которая встроена в TIA Portal и старой WinCC - WinCC Flexible 2008, а также сделанной на ее основе WinCC Flexible Smart.

Короче, суть вот в чём: старые WinCC Flexible разрешали как передачу текущих даты/времени, установленных на панели оператора, в память ПЛК, так и, одновременно с этим, разрешали автоматическое обновление даты/времени панели оператора через считывание. Та WinCC, которая теперь находится внутри TIA Portal, позволяет делать только что-то одно из перечисленного - либо синхронизировать HMI с PLC, либо передавать дату/время HMI на PLC, но не одновременно. Иначе будет вот что:


Почему ограничение введено - совершенно не имеет объяснения, потому что, с точки зрения HMI, значения даты/времени - это обычные данные для приёма-передачи, такие же теги, т.е. вы можете создать хоть сотню тегов, которые как угодно друг с другом переплетаются, привязаны к одним и тем же (или частично) адресам памяти, настроены на разные ячейки на чтение, запись или и то, и другое одновременно - никаких ограничений нет - но почему-то с датой/времени Siemens вдруг решил ввести ограничения. В любом случае это особо не повлияет на нашу настройку, далее я покажу в чём будет заключаться отличие.

Резервирование памяти PLC


На PLC необходимо создать блок данных и ячейки в нём, где будут хранится:

1. Job MailBox панели оператора.
Это та ячейка, которая будет давать команду HMI на пересылку её текущих даты/времени в PLC. Как только значение MailBox станет равно 40, HMI выполнит команду, а MailBox сбросит в 0.

2. Дата и время HMI в формате BCD.
Это ячейки памяти PLC, куда будут сохранены дата и время HMI. Представлены они в двоично-десятичном формате (BCD), причём каждое число в виде 1 байта, включая год. А тот самый год, который вроде как измеряется в тысячах, выглядит либо как число от 80 до 99 - это будет означать соответственно с 1980-го по 1999 год, либо как число от 0 до 29 - это будет год с 2000 по 2029-ый. Вот так это представлено в справке как WinCC Flexible:


В моём примере ниже ячейки памяти для даты/времени HMI будут обозначены как HMI2_DATETIME.

3. Дата и время тех же значений HMI, что и в предыдущем пункте, но переведённых в виде нормальных чисел.
В моём примере ниже - HMI_DATETIME.

4. Дата и время PLC в обычном формате.
В таком формате они возвращаются системной функцией PLC.
В моём примере ниже - PLC_DATETIME.

5. Дата и время PLC в формате BCD.
В таком формате они принимаются HMI для обновления её значений
В моём примере ниже - PLC2_DATETIME.


6. "Новые" дату и время для обновления на PLC.
Т.е. на HMI создаются поля для ввода даты и времени, привязанные к этим ячейкам PLC, и кнопка, по нажатию которой информация из этих ячеек передаётся в системную функцию обновления даты/времени PLC. В моём примере ниже - NEW_DATETIME.

7. Биты для управляющих кнопок:
- скопировать текущие дату/время HMI в ячейки для новых даты/времени
- скопировать текущие дату/время PLC в ячейки для новых даты/времени
- применить новую дату/время PLC (кнопка, упомянутая в пункте 6)

В общем и целом зарезервированная память в ПЛК выглядит так:


Исходный код на PLC


// Считать время HMI и отображать его в отдельных ячейках HMI
IF(#Front) THEN
   "HMI_DB".HMI_MAILBOX.Value := 40;
END_IF;

"HMI_DB".HMI_DATETIME.Year  := BCD16_TO_INT("HMI_DB".HMI2_DATETIME.Year);
"HMI_DB".HMI_DATETIME.Month := BCD16_TO_INT("HMI_DB".HMI2_DATETIME.Month);
"HMI_DB".HMI_DATETIME.Day   := BCD16_TO_INT("HMI_DB".HMI2_DATETIME.Day);
"HMI_DB".HMI_DATETIME.Hour  := BCD16_TO_INT("HMI_DB".HMI2_DATETIME.Hour);
"HMI_DB".HMI_DATETIME.Min   := BCD16_TO_INT("HMI_DB".HMI2_DATETIME.Min);
"HMI_DB".HMI_DATETIME.Sec   := BCD16_TO_INT("HMI_DB".HMI2_DATETIME.Sec);

// Записать заданное с HMI время в PLC

IF("HMI_DB".HMI_btn_Clock_Set_to_PLC) THEN
   "NotUsedInt" := WR_SYS_T(IN := "HMI_DB".NEW_DATETIME);
   "HMI_DB".HMI_btn_Clock_Set_to_PLC := false;
END_IF;

// Считать время PLC, отображать его в отдельных ячейках HMI
// и отправлять его на панель для обновления времени HMI (через PLC2_DATETIME)

"NotUsedInt" := RD_SYS_T(OUT => "HMI_DB".PLC_DATETIME);
"HMI_DB".PLC2_DATETIME.Year := "BCD_TO_BYTE"("HMI_DB".PLC_DATETIME.YEAR);
"HMI_DB".PLC2_DATETIME.Month := "BCD_TO_BYTE"("HMI_DB".PLC_DATETIME.MONTH);
"HMI_DB".PLC2_DATETIME.Day := "BCD_TO_BYTE"("HMI_DB".PLC_DATETIME.DAY);
"HMI_DB".PLC2_DATETIME.Hour := "BCD_TO_BYTE"("HMI_DB".PLC_DATETIME.HOUR);
"HMI_DB".PLC2_DATETIME.Min := "BCD_TO_BYTE"("HMI_DB".PLC_DATETIME.MINUTE);
"HMI_DB".PLC2_DATETIME.Sec := "BCD_TO_BYTE"("HMI_DB".PLC_DATETIME.SECOND);     

// Скопировать текущее время PLC в NEW

IF("HMI_DB".HMI_btn_Clock_Copy_PLC_to_NEW) THEN
   "HMI_DB".NEW_DATETIME.YEAR := "HMI_DB".PLC_DATETIME.YEAR;
   "HMI_DB".NEW_DATETIME.MONTH := "HMI_DB".PLC_DATETIME.MONTH;
   "HMI_DB".NEW_DATETIME.DAY := "HMI_DB".PLC_DATETIME.DAY;
   "HMI_DB".NEW_DATETIME.HOUR := "HMI_DB".PLC_DATETIME.HOUR;
   "HMI_DB".NEW_DATETIME.MINUTE := "HMI_DB".PLC_DATETIME.MINUTE;
   "HMI_DB".NEW_DATETIME.SECOND := "HMI_DB".PLC_DATETIME.SECOND;
   "HMI_DB".HMI_btn_Clock_Copy_PLC_to_NEW := false;
END_IF;

// Скопировать текущее время HMI в NEW

IF("HMI_DB".HMI_btn_Clock_Copy_HMI_to_NEW) THEN
   "HMI_DB".NEW_DATETIME.YEAR := BYTE_TO_UINT("HMI_DB".HMI_DATETIME.Year) + 2000;
   "HMI_DB".NEW_DATETIME.MONTH := "HMI_DB".HMI_DATETIME.Month;
   "HMI_DB".NEW_DATETIME.DAY := "HMI_DB".HMI_DATETIME.Day;
   "HMI_DB".NEW_DATETIME.HOUR := "HMI_DB".HMI_DATETIME.Hour;
   "HMI_DB".NEW_DATETIME.MINUTE := "HMI_DB".HMI_DATETIME.Min;
   "HMI_DB".NEW_DATETIME.SECOND := "HMI_DB".HMI_DATETIME.Sec;
   "HMI_DB".HMI_btn_Clock_Copy_HMI_to_NEW := false;
END_IF;


Настройки HMI


В HMI необходимо зайти в раздел Connections, а затем перейти во вкладку Area Pointer, где есть общая часть (For all connections), связанная с обновлением значения даты/времени HMI (синхронизация может происходить только с одним PLC) и конкретная часть для каждого PLC отдельно (For each connection) - в частности, сохранение текущей даты/времени HMI в PLC:


Следовательно, для нашего примера на PLC настройки HMI будут выглядеть следующим образом:


Теперь посмотрим, как это выглядит на панелях оператора Siemens Smart:


Как видно, номера строк не совпадают для одних и тех же параметров английской версии WinCC Flexible 2008 и WinCC Flexible Smart (и WinCC Flexible China SP4), однако, тут легко догадаться, какие строки нам нужны в китайской версии, даже не понимая иероглифов: четвёртая колонка Length указывают длину параметра (в 16-битных словах), которая для всех параметров каждого раздела разная. Т.е. тут легко сопоставить длины параметров:


Далее необходимо добавить все наши данные в таблицу тэгов.


Ну, и создаём экран такого плана:


Надеюсь, не надо объяснять, какой тэг к какой ячейке и к какой кнопке привязывается, тут и так всё очевидно. Единственное, следует обратить внимание, что год в считываемой с HMI текущей дате в моём примере состоит из двух ячеек: одна просто декоративная, в которой постоянно светится число "20", а вторая часть - это двузначное число HMI_DATETIME.Year. В принципе, можно было сделать и по-другому, переводя считываемое с HMI значение в полноценное четырёхзначное число (в функции на PLC).

WinCC в TIA Portal


Тут всё почти тоже самое, кроме того, что мы не можем считывать с HMI текущие дату/время, поскольку мы должны выбирать: либо задаём время на HMI, либо считываем его. Иначе получаем такую ошибку:  

The area pointer 'Date/time' is invalid. The "Date/Time" and "Date/Time PLC" global area pointers cannot be set at the same time

Поскольку у нас основная задача как раз задавать время, то функцию считывания мы не используем. Это не смертельно, хотя, повторюсь, совершенно непонятно, зачем Siemens ввёл такое ограничение. Соответственно, мы тут тоже настраиваем раздел Connections -> Area Pointer , но в усечённом виде, поскольку задаём только параметр Date/Time PLC. Ну, и Siemens не был бы Siemens'ом, если бы в этой версии WinCC не поставил всё с ног на голову:


Т.е. заметьте, что разделы и называются по другому: тот, который был раньше For all connections, теперь зовётся Global area pointer of HMI device, а раздел For each connection тут и вовсе не имеет названия. Ну, и конечно, разделы поменяны местами, как же без этого. Поменяны местами и строки: интересующая нас единственная строка Date/Time PLC теперь последняя в списке.

Соответсвенно, на экране больше нет ячеек для считанных с HMI значений и нет кнопки для копирования значений HMI в ячейки для задаваемых даты/времени:


Всё.




пятница, 22 мая 2020 г.

Особенность настройки ICP CON 7061D


Сегодняшняя тема - про модули ввода-вывода от фирмы ICP DAS. Долгое время у меня были сомнения насчёт использования различного рода дешёвых модулей ввода-вывода в связке с контроллером S7-1200, но вот возник объект, куда клиентом были закуплены модули M-7061D - дискретные выходы (12 DQ) и M-7041D - дискретные входы (14 DI).

По умолчанию на этих модулях используется протокол DCON. Мы с этим протоколом не работаем, мы используем Modbus RTU. Для изменения протокола и настройки параметров связи необходимо установить программу DCON Utility Pro. Где скачивать ПО и как настраивать оборудование, объясняет инструкция, которая идёт в комплекте с оборудованием. Инструкция - общая для любых модулей I-7000 и M-7000.


В левой верхней части мы видим настройки по умолчанию (Default Communication Parameters). В частности, там сказано, что для модулей M-7000 протоколом по умолчанию является Modbus RTU с адресом 1 и параметрами 9600 kbps, 8E1. Я не уверен, что это так. У меня было в распоряжении несколько модулей M-7061D и M-7041D и, по крайней мере некоторые из них, будучи в состоянии "с завода", при поиске устройств сети (поиск по обоим протоколам - DCON и Modbus RTU) вообще никакого результата не показали (может быть, надо было искать Modbus ASCII, - ну, я не пробовал). Поэтому я дальше этот поиск осуществлять не стал, а сразу перевел все устройства в режим INIT, и дальше уже с ним вёл настройку.

На модулях M-7041D режим INIT активируется тумблером INIT/NORMAL в задней части модуля.
На модулях M-7061D активация происходит перемычкой с клеммы GND на клемму INIT*.
Зачем так по-разному сделано? Я так понимаю, что изначально они хотели везде сделать активацию перемычкой, но на модуле M-7061D сравнительно много клемм, потому что на нём каждый канал дискретного выхода является отдельной группой (т.е. каждый канал использует две клеммы). Ну, соответственно, клемм получилось много, а размер самого устройства стандартный - как для всех M-7000, в итоге клемме INIT* места не досталось, вот и придумал производитель сделать тумблер. Как по мне, ни один из способов перехода в режим INIT не является удобным, потому что тумблер на M-7061D расположен в задней части модуля. Это значит, что если вы приедете на объект с уже собранным шкафом автоматики и установленными в нём на din-рейках устройствами, то каждое из них придётся снимать, чтобы переключать тумблер на обратной стороне. Перемычка - тоже так себе вариант. Почему не сделать было тумблер на передней части модуля (да и вообще всю настройку параметров интерфейса тумблерами) - неизвестно.

Далее под пунктом 2 мы видим рекламу конвертеров RS232/RS485 и USB/RS485 от ICP, которые они рекомендуют использовать для подключения к компу. Пункт 3 - это ссылка на скачивание DCON Utility Pro.

Переходим к пункту 4, тут мы видим картинку последовательности действий для поиска устройства:


Непонятно, зачем они понаставили галочек на разные скорости и на разные протоколы, если перед этим уже перевели устройства в режим INIT. В этом режиме искать надо на скорости 9600 kbps, протокол - DCON, без проверки контрольной суммы (Checksum Disabled), формат - без проверки на чётность, 8 бит, 1 стоп-бит ("N,8,1").

Переходим к пятому пункту. Тут у нас настройка параметров связи.

Ну, т.е. видите: (1) - нажали кнопку поиска, (2) - сделали двойной щелчок мыши по названию устройства в первой колонке таблицы, открылось окно настройки, (3) - поменяли всё, что нужно, (4) - нажали кнопку "Set...",  (5) - на выход, настройка закончена.

Для модуля M-7041D у меня всё так и настроилось. Вот смотрим нотариально заверенный скриншот:


А вот с модулями M-7061D что-то пошло не так:


В общем-то пляска с бубном привела к нажатию вот такой вот кнопочки:


И, собственно, появилось окошко, где в выпадающем списке можно выбрать тип модуля:


Выбираем M-7061D, после чего открывается окно настройки интерфейса - похожее на то, которое было до этого, но в заголовке окна присутствует надпись Offline Configuration. и кнопка другая - Write Configuration to I/O Module. Тут всё настроилось без проблем:


Далее обесточиваем модуль, убираем INIT, включаем и всё работает.

Регистры Modbus RTU


Для считывания текущего состояния дискретных входов модуля M-7041D используются Modbus-функция 02: Read input bits. Регистр для чтения - 10001. Функция позволяет читать несколько битов подряд за один запрос, т.е. читаем сразу 14 битов.

Для записи значений дискретных выходов модуля M-7061D используется Modbus-функция 05: Writing an output bit (если надо записать 1 бит) или функция 15: Writing multiple output bits - если надо записать значения для нескольких выходов (пишем сразу все 12 выходов, например). Регистр для записи - 1.

Возможно, что будет желание проверять состояние связи с каждым модулем (для индикации ошибки) через считывание контрольного слова. Я использовал для этого чтение значения из регистра 40485. В этом регистре хранится текущий адрес устройства в сети Modbus, это значение условно можно считать проверочным значением для текущего устройства.

P. S.
В общем-то, ничего пока сказать о качестве этих модулей не могу. Но думаю, что шансов, что они будут нормально работать, гораздо больше, чем у модулей Овена и других подобных горе-производителей. Ценник более чем приемлем, хотя вся эта приемлемость проистекает напрямую из-за возможности покупать модули тут, в России через юрлицо, не занимаясь обналом. Если же сравнивать с ценами Aliexpress, то я бы предпочел купить S7-200 и noname-модули к нему, и так же по Modbus сделать связку с S7-1200, потому что на S7-200 можно переложить часть функционала, включая проверки на дребезг контактов и масштабирование аналогового сигнала. Ценник на модули ICP CON у официалов явно завышен, это легко проверить, если сравнить с ценами на них же на Aliexpress, хотя модулей M-7000 (Modbus) я с ходу не нашел, но вот, например, модуль I-7000 для подключения 8 термопар (I-7019R) на Aliexpress стоит $60, а в России - $250, на дискретные модули разбег, понятное дело, меньший (раза в полтора - два), но просто из-за невозможности торгашей сделать конкурентную цену высокой.

четверг, 13 февраля 2020 г.

Ошибка OPCENUM в Windows 10

В Windows 10 при попытке запустить приложение, написанное с использованием WTCLIENT.DLL, выдало ошибку:

Failed to Execute OPCENUM

На Windows 7 подобной ошибки никогда не возникало. Ну, короче, необходимо в таком случае установить OPC Core Components.

Качаем тут

или на официальном сайте - но там необходимо сначала зарегистрироваться...

P.S. Напомню, WTClient - удобная бесплатная библиотека для написания приложений с использованием OPC-сервера, только качать ее надо тут