Хотя цифровые операции ввода-вывода позволяют решать широкий круг задач, однако наличие в микроконтроллере платы Arduino встроенного аналого-цифрового преобразователя (АЦП) и возможность вывода аналоговых сигналов с помощью широтно-импульсной модуляции (ШИМ) обеспечивают работу с аналоговыми датчиками и всевозможными исполнительными устройствами, воздействующими на объект пропорционально управляющему сигналу.
Строго говоря, в режиме вывода все линии портов Arduino могут передавать только дискретные сигналы, имеющие лишь два состояния. Но микроконтроллер способен изменять эти состояния очень быстро, генерируя прямоугольные импульсы. Если подать эти импульсы на какое-либо устройство, обладающее инерционными свойствами, то оно станет вести себя так, будто подаваемое на него напряжение постоянное, равное среднему значению импульсного, и меняется плавно, а не скачками между высоким и низким логическими уровнями.
В режиме ШИМ порт формирует импульсный сигнал постоянной частоты и переменной скважности (это отношение периода следования импульсов к их длительности). Часто вместо скважности оперируют обратной ей величиной - коэффициентом заполнения, который можно изменять от 0 (нет импульсов) до 100% (импульсы следуют, слившись, без пауз). Следовательно, хотя в каждый отдельный момент выходное напряжение соответствует высокому или низкому логическому уровню, его среднее значение пропорционально коэффициенту заполнения. Если к этому выходу подключить обычный мультиметр, он покажет это значение (конечно, если частотаимпульсов достаточно высока).
В Arduino UNO в режиме ШИМ могут работать выходы D3, D5, D6, D9, D10 и D11. Обычно на плате они помечены знаками "~" или аббревиатурами "PWM". Следует заметить, что у плат Arduino других модификаций число таких выходов может быть больше или меньше.
В простейшем случае ШИМ можно применить для управления яркостью свечения светодиода. Этот прибор практически безынерционен, но человеческое зрение обладает достаточной инерционностью, чтобы последовательность быстрых вспышек светодиода воспринималась как непрерывное свечение с зависящей от коэффициента заполнения яркостью.
Дискретные выходы, способные формировать ШИМ, настроены на использование этого режима по умолчанию, поэтому вызывать функцию pinMode() для их работы в таком режиме не нужно. Для установки коэффициента заполнения ШИМ-сигнала имеется стандартная функция analogWrite(N, M), где N - номер вывода, M - число, пропорциональное требуемому коэффициенту заполнения. Оно должно лежать в интервале от 0 до 255, причём 0 соответствует нулевому коэффициенту заполнения (навыходе постоянный низкий уровень), 255 - коэффициенту заполнения 100 % (на выходе постоянный высокий уровень). Временные диаграммы выходного напряжения при некоторых значениях M и соответственно коэффициента заполнения Кз показаны на рис. 1.
Рис. 1. Временные диаграммы выходного напряжения
Для примера рассмотрим приведённую в табл. 1 программу, которая постепенно увеличивает яркость свечения светодиода, подключённого к цифровому выходу D9, а затем постепенно уменьшает её. Она основана на стандартном примере examples3.AnalogFading из комплекта поставки Arduino IDE. Перебор значений коэффициентазаполнения импульсов реализован здесь с помощью уже рассмотренных в [1] операторов цикла for.
Таблица 1.
Для приёма аналоговых сигналов от внешних устройств в Arduino предназначены входы A0-A5, по умолчанию установленные в нужное для этого состояние, так что дополнительной инициализации не требуется. АЦП, встроенный в Arduino UNO, формирует 10-разрядные двоичные коды и входное напряжение, лежащее в интервале от 0 до +5 В, преобразует в целое число от 0 до 1023 (210-1).
Для считывания результата преобразования служит функция analogRead(N), где N - номер аналогового входа.
К аналоговым входам Arduino можно подключать разнообразные датчики, выходное напряжение которых пропорционально измеряемой величине (переменные резисторы, терморезисторы, фоторезисторы и др.). Однако нужно помнить, что на аналоговый вход можно подавать напряжение лишь от 0 до +5 В. Если выходное напряжение датчика лежит в другом интервале или оно отрицательной полярности, сигнал необходимо предварительно уложить в указанный интервал. Опрос аналогового входа выполняется с частотой менее 10 кГц [2], что может оказаться недостаточным для анализа некоторых быстроизменяющихся сигналов.
Наличие аналоговых входов позволяет превратить Arduino в простейший цифровой вольтметр, измеряющий постоянное напряжение от 0 до +5 В и передающий результат измерения в компьютер. Для этого достаточно загрузить в Arduino программу, приведённую в табл. 2.
Таблица 2
Обратите внимание, что в программе константами заданы образцовое напряжение АЦП Uref (в милливольтах) и коэффициент пересчё-тавыходного кодаАЦП в напряжение Ku. Значение коэффициента вычисляется делением заданного образцового напряжения на 1023. Коэффициент обычно дробный, поэтому константа Ки имеет тип float (число с плавающей запятой). Константа Uref имеет такой же тип для правильного вычисления коэффициента. Поскольку в правой части формулы находятся только константы, вычисляет коэффициент не микроконтроллер, выполняя программу, а сам компилятор на этапе её трансляции.
Всё это позволяет повысить точность вольтметра, измерив мультиметром точное значение образцового напряжения на выводе Uref платы Arduino и записав его в программу, присвоив константе Uref. О других способах повысить точность аналого-цифрового преобразования можно прочитать в [3, 4].
При работе рассматриваемой программы на плате мигает светодиод TX, сигнализирующий о передаче информации через последовательный порт. Светодиод RX не светится, так как компьютер ничего не передаёт в ответ. Встроенный терминал Arduino IDE отображает принятую информацию (рис. 2) - результаты измерения напряжения гальванической батареи 3332.
Рис. 2. Окно программы
Arduino может подавать не только световые, но и звуковые сигналы. Для этого к одному из его выходов необходимо подключить пьезоизлучатель звука, например ЗП-1 (рис. 3).
Рис. 3. Подключение пьезоизлучателя звука
Для работы со звуком предусмотрена специальная функция tone(N, F, T), где N - номер вывода, на котором будут сформированы прямоугольные импульсы; F - частота звука, Гц; T - длительность звука, мс. Последний параметр не обязателен. В его отсутствие звук будет непрерывным. Чтобы выключить его, предусмотрена функция noTone(N).
Конечно, пьезокерамический излучатель звука сложно назвать устройством высококачественного воспроизведения, а формируемый микроконтроллером сигнал имеет прямоугольную форму, тем не менее использование этих функций позволяет исполнять несложные мелодии. Пример приведён в табл. 3. Это немного изменённая программа examples 02.Digital oneMelody, входящая в комплект среды разработки Arduino IDE. Поскольку задавать вручную частоту каждой ноты мелодии неудобно, к программе в её заголовке директивой #include подключён файл pitches.h. Эта операция равносильна включению в программу полного текста этого файла. В рассматриваемом случае он содержит список названий нот, которые можно воспроизвести, и их частот.
Таблица 3
Излучатель звука должен быть подключён к выходу D8.
Для программы мелодия - это последовательность однотипных констант (значений частоты), которые удобно объединить в массив - пронумерованный список однотипных элементов. При объявлении массива нужно либо перечислить все его элементы, либо указать их общее число. Учтите, что нумерация элементов массива всегда начинается с нуля.
В рассматриваемом примере использованы два массива: int melody[] содержит названия нот мелодии, int note Durations[] - их длительность в миллисекундах. Для обращения к элементу массива указывают его имя с заключённым в квадратные скобки порядковым номером. Чтобы иметь возможность легко менять число нот в мелодии, оно вычисляется с использованием функций sizeof(V), возвращающих число байтов, занимаемых её аргументом (переменной или их массивом) в памяти микроконтроллера. В рассматриваемом случае массив melody занимает 16 байт, а длина его элементов типа int - два байта. Поэтому переменная Note получает значение 8 и именно столько раз будет повторено тело цикла for, поочерёдно воспроизводящее ноты.
Если к массиву melody[] добавить несколько нот, соответственно изменится и значение Note. Нужно только не забыть дополнить массив noteDurations[] длительностями звучания этих нот.
Поскольку мелодия исполняется один раз, все необходимые для этого операции помещены внутрь функции setup().
Для повторного исполнения нужно привести микроконтроллер в исходное состояние, нажав на находящуюся на плате Arduino кнопку RESET
Рассмотренные в статье программы для Arduino можно скачать здесь.
Литература
1. Лекомцев Д. Arduino. Операции цифрового ввода-вывода. - Радио, 2016, № 8, с. 51-54.
2. Аналоговые измерения с Arduino. - URL: http://robotosha.ru/arduino/analog-measurements-arduino.html (02.06.16).
3. Arduino Language Reference. Analog I/O - analogReference(). - URL: https://www. arduino.cc/en/Reference/AnalogReference (02.06.16).
4. Функция analogReference(). - URL: http:// arduino.ru/Reference/AnalogReference (02.06.16).
Автор: Д. Лекомцев, г. Орёл