Как принять сигнал с приемника в AVR (PWM|Digital)

Петруччо

Коллеги, позвольте задать дилетантский вопрос:

сейчас встала необходимость принять на мегу сигнал с приемника, на этом канале висит несколько серв. Идея такая:

  1. Принимаем сигнал на ICP, меряем длительность импульса счетчиком.
  2. Значение длины полученного импульса программно корректируется, т.е. от него либо отнимается время, либо сколько-то прибавляется.
  3. формируется новый импульс заданной длинны на соответствующем PWM пине проца.

Таким образом имея 1 вход можно на него через проц повесить 2-3-4 серво в режиме секвенсора.

Вопрос собсно такой: принять и обсчитать импульс это не проблема, новый тоже сформировать нет, но насколько я понимаю пока проц будет “думать” над текущим обрабатываемым импульсом в это время на ICP будет тикать новый и его мега не успеет уже принять и обработать. Т.е. если я правильно понимаю как минимум будет запаздывание в 1 цикл ШИМ от приёмника. Можно как-то решить эту проблему или дело настолько серьезно, что напрашивается RTOS?
Для справки, JR matchbox выполняет именно такую ф-ию, но в руках не держал, не разбирал… Если я прав, то это в принципе можно реализовать на 2-3-4 независимых камнях типа тиньки25 или 13, у которой есть входы ICP.

Частота 4 МГц, камень ATMega32…пока. На нем отрабатываются узлы, потом все переезжает в 128ю.

Буду признателен за любые подсказки.С Уважением.

V_Alex

Маскимальная длительность пришедшего с приемника импульса - 2 миллисекунды. Такт поступления импульсов - 20 миллисекунд. Итого остается 18 милисекунд.
На последовательную генерацию трех импульсов (максимум те-же 2 миллисекунды каждый) уйдет 6 миллисекунд. Итого, на все расчетные удовольствия остается 12 миллисекунд.
Вторичные импульсы есс-но будут отставать от исходного, но все будут сгенерированы до прихода следующего импульса с приемника.

Петруччо

Благодарю!
Т.е. генерация новых “обсчитанных” импульсов идет в процессоре в фоновом режиме, в то время как проц обрабатывает новые входящие, правильно?
Извините за нескромность, у Вас часом нет этого куска кода? Мне хоть посмотреть в какую сторону бежать…😊

V_Alex

Камень - тини 13А, частота внутреннего RC-генератора 9,6 МГц. Импульс снимается в прямом цикле без прерываний с 4 входа порта В. После метки main10 в регистрах Delay находится разность между 2,2 мс и фактической длиной импульса. Импульс меряется тактами по 6 тиков в каждом. Принцип последующей генерации выходных импульсов после обсчета - такой-же.
=============================================
;********** переопределям регистры ***********************
.def Delay_L = R19 ; задержка (младший байт)
.def Delay_H = R20 ; задержка (старший байт)

;********* задаем константы (9,6 МГц, 6 тактов) **********
.equ VRRST = 3520 ; верхний порог (2,200 мс)

;********** обработка импульса ********************************
ldi Delay_L,Low(VRRST) ; начинаем обработку импульса
ldi Delay_H,High(VRRST); загружаем 2,2 мс в регистры задержки
pul_3: sbis PinB, 4
rjmp main10 ; выходим, если импульс закончился
subi Delay_L, 1
sbci Delay_H, 0
brcc pul_3
main10: …
=======================================
Сразу предупреждаю, что в плане ассемблера я “начинающий” программист.
Возможно кто-то из гуру расщедрится на более грамотное решение.

Петруччо

Спасибо! 😃
Ну, я “сишник”, за ассемблером к отцу пойду, он у меня спец по нему. В принципе всё понятно, попробую это на си портировать, но чувствуется без прерывания не обойтись.

V_Alex

Код делался под примитивную задачу, хотелось обойтись без прерываний.

Петруччо

Александр, а не поясните пжлст: что значит

V_Alex:

Импульс снимается в прямом цикле

? Т.е. как я понимаю программа в бесконечном цикле (jmp loop) проверяет изменение сигнала на входе (1 или 0) и уже потом обрабатывает его?

RL2000
Aleksey_Gorelikov:

Не проще запретить внешнее прерывание?

Такой способ не подойдет, задумал вообще не считывать PWM-сигнал с приемника. Сразу с приемника на сервопривод, а в нужный момент отобрать линию у приемника. Разве что ставить дополнительную микросхему и передавать тщательно вычисленную длительность через UART или еще как.

Петруччо
RL2000:

Сразу с приемника на сервопривод

RL2000:

отобрать линию у приемника

RL2000:

дополнительную микросхему и передавать тщательно вычисленную длительность через UART

это как??😃 В любом случае придется процессор в разрыв тогда ставить.

V_Alex
Петруччо:

… как я понимаю программа в бесконечном цикле (jmp loop) проверяет изменение сигнала на входе (1 или 0) и уже потом обрабатывает его?

Именно так. После инициализации запрыгиваем в бесконечный цикл. Если на входе 4 установилась единица, значит там сервоимпульс. Вот его и меряем. Замер заканчивается с окончанием импульса и наступлением синхропаузы (на входе 4 - ноль). Дальше делаем с импульсом все, что нам надо и, по аналогичной замеру методе, выводим его на любую другую лапку камня. Можно еще раз обработать и вывести на еще одну лапку.
Главное - уложиться в синхропаузу (18 мс). Обработали, вывели, расслабились. Ждем, пока на входе 4 не закончится ноль. Как закончился, перепрыгиваем к обработке входного импульса и все повторяется по-новой.
По хорошему перед первым замером пару импульсов лучше пропустить, чтобы не влезть в середину импульса. После замера можно проверить импульс на попадание в допустимые пределы.

Петруччо
V_Alex:

Именно так.

Все гениальное простоB-)
Спасибо, попробую повторить оное на сях в меге.

Vladimir_N

На тиньке 13 по другому сделать и не получится у нее нет 16 битного таймера и входа ICP. А вот на меге желательно использовать встроенные аппаратные ресурсы.
Вначале вы правильно решили вводить импульс при помощи ICPи выводить ШИМ.
Ввод и вывод в фоновом режиме под прерываниями, между ними обработка импульса.
При этом джитер измерения и вывода импульса минимальный и на обработку остается вагон времени.
Есть несколько тонкостей чисто аппаратных.
Советую использовать ICPвход не напрямую, а через встроенный аналоговый компаратор, у меги есть функция их сцепки. В этом случае исчезнут проблемы с уровнями сигнала от приемника.
Выводить импульс удобнее при помощи прерывания Compare соответствующего выхода.
Кварц удобней использовать на 8 МГц тогда при установке делителя 16 битного таймера счетчика на 8. Мы получаем 1 тик таймера =1 микросекунда. Проще будет потом производить расчеты в программе.
Счетчик не перезагружаем, пусть молотит по кругу. Все измерения импульсов производим как разность показаний счетчика в начале и в конце импульса.

Петруччо

ага, спасибо огромное за науку! В принципе я так и думал что так правильно, просто терзали сомнения успеет ли камень обрабатывать новый входящий импульс, в то время как PWM генерит текущий обсчитанный. про компаратор - посмотрю осциллографом что там на выходе, м.б. и обойдусь малой кровью, компаратор может еще понадобиться, так что можно кинуть в принципе какой-нить простенький ОУ на входе.

Probelzaelo
project_Ikar:

что такое по Вашему “аппаратно-заточенный под ШИМ вход” и что за чудо схемотехника стоит за этим входом, что он носит гордое название “аппаратно-заточенный”?..

Полагаю, ув. Blade, просто хотел сказать что принимать и обрабатывать ШИМ сигналы гораздо удобнее если использовать внешние аппаратные прерывания, вот и вся “заточка” справедливо это для 4 и 5 (но не 5 и 6 ) ног Атмеги8,168, 328, в Ардуино как раз, и чаще всего, именно два последние и используются, это внешние прерывания 0 и 1, . Действительно, для ввода ШИМ сигналов гораздо удобнее пользоваться именно этими портами Атмеги. Учитывая возможность использовать обработчик прерывания по любому(каждому) изменению логического состояния сигнала на входе, сам бог велел этим воспользоваться в своих целях )) Очень удобно, оказавшись в обработчике прерывания первым делом проверить состояние входа и в зависимости от того чем вызвано прерывание началом или окончанием импульса либо если начало то обнулить и запустить таймер, либо, если окончание, то прочесть из таймера и вычислить вполне точное значение длительности входного импульса… остальными “ногами” тоже вполне можно пользоваться для ввода ШИМ но тогда будет маловато времени для какой либо другой деятельности процессора, а его вполне можно бы использовать и с бОльшей пользой.

V_Alex

Да, это более правильный подход, но в жизни бывают ситуации, когда 16-разрядный таймер нужен для других целей. ИМХО для 90 % модельных приложений Мега не нужна. Очень хороши 24 и 25 тиньки.

Probelzaelo
V_Alex:

Попробую объяснить свою точку зрения. Рабочие коды коммерческих версий устройств выкладывать никто не будет. Project Ikar звучил это в соседней теме совершенно определенно. Я считаю это справедливым, поскольку затраты времени и здоровья на вылизывание софта могут быть весьма значительными.

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

V_Alex
Probelzaelo:

Давайте определимся с тем что такое коммерческая версия устройства…

А чего тут определяться? Ваша точка зрения никак не противоречит моей. Каждый сам решает, как поступать со своими разработками. У меня иногда получается одновременно и делиться: www.rcdesign.ru/articles/…/battery_indicator и
наживаться: igva.ru/prd/led/10led/10led_1.htm, чего и Вам желаю 😃.

Probelzaelo
V_Alex:

У меня иногда получается одновременно и делиться: www.rcdesign.ru/articles/…/battery_indicator и
наживаться: igva.ru/prd/led/10led/10led_1.htm, чего и Вам желаю 😃.

Это правильно, но на мне в той же ситуации нажились китайцы )) hobbyking.com/…/__6589__Battery_Monitor_2_6S.html
впрочем я не проиграл, как то в цифрах понятнее, а с другой стороны при установке на борту, цвета различать даже проще чем циферки.
Но за вас я, в любом случае, очень радуюсь … ))

Boroda

Гланды конечно можно разными способами удалять…
В семействе восьмых мег есть превосходный механизм для аппаратного захвата и генерации шим. Даже представить не могу, как ещё можно использовать в модельном применении 16-бит таймер, кроме как для безджиттерного измерения и генерации сервосигналов? Есть выбор входа с цифровой ноги или с аналогового компаратора, есть подавление шума, есть регистр захвата значения таймера, есть гибкая генерация прерываний и флагов, есть механизм коррекции фазы и частоты по ICR для генерации PWM. И всё это аппаратное! За 48-ю мегу нынче просят денег меньше чем за тиньку. Так нет-же, люди продолжают искать джиттер и конфликты прерываний на свою голову!

harrystar
RL2000:

Правда, при попытке выдать принятый PWM на выход (через 16-битный таймер, т.к. основная программа занята), серва дергается. Дискретизация по времени приема PWM с сервотестера - порядка 52мкс, если ставить больше, как писали - 2000 тиков таймера на 2мс, то на основную программу просто не остается тактов процессора.

Складывается ощущение, что вы каждые 52 мкс (без всяких условий) уходите в прерывание, считываете состояние входа и ставите его на выходе.
Конечно будет дергаться, причем будет жрать много ресурсов, чем точнее, тем больше.
Алгоритм то совсем другой. Нужно сделать прерывание не по таймеру, а по изменению состояния входа. Соотв при старте импульса записывать значение таймера (который тикает примерно 2000 тиков или больше на 2 мс), а при окончании импульса вычитать записанное значение из текущего. Т.о. прерывание должно происходить всего 2 раза за 20 мс (100 раз в секунду) и на основную программу остается куча времени.

V_Alex
Boroda:

Гланды конечно можно разными способами удалять…
Даже представить не могу, как ещё можно использовать в модельном применении 16-бит таймер, кроме как для безджиттерного измерения и генерации сервосигналов?..

Можно, можно 😃 . Все зависит от конкретной задачи. Тут полстраницы ушло на обсуждение, как лучше организовать переключение сигнала с двух разных источников на одну серву. Правильный ответ - коммутатор на одном корпусе КМОП-логики за пять рублей. Считаем дальше: 13 тинька стоит 0,5-0,6 бакса. РУчная пайка одной ноги при мелкой серии - 2 цента и выше. И размер посадочного места (стоимость плате) тоже имеет значение. Если строить для себя в одном экземпляре - нет проблем (собственно с этого тема и начиналась). Если серию, то значение имеет каждый цент. Кусочек кода, который я выкладывал, из серийного изделия. Импульс обрабатывается в 16-разрядном режиме, соответственно джиттера нет. Прерывание там только одно - по WDT, других нет, соответственно нет и конфликтов. А 16-разрядный таймер бывает нужен, например, для организации 9-10 разрядного ШИМа, типа чтобы свечку от 12 Вольт калить. Кстати, в 24 тиньке он есть.