В этой статье мы сделаем акцент на портах ввода/вывода МК ATtiny2313 и заставим их работать на выход. То есть ножка микроконтроллера (МК) будет выдавать сигнал либо логической единички, либо нуля. Приказывать, что выдавать на ножку, будем мы сами, а точнее, программа, которую мы с вами разработаем.
Все начинали изучать AVR c мигания светодиода. Мы даже делали целую гирлянду из 4 светодиодов в этой статье. Но что, если 4 светодиодов нам мало, и мы хотим чего-то большего? Ну раз хочется, то давайте сделаем!
Выбор микроконтроллера
«Подопытным кроликом» у нас будет старый добрый МК ATtiny2313
Долго не думая, находим в интернете на него даташит и смотрим цоколевку
Так как я хочу навесить на него кучу светодиодов и сделать из них что-то подобие бегущей дорожки, для нас важно рассмотреть, какие пины (выводы МК) позволят мне это сделать. Что мне здесь надо найти? Это, конечно же, порты ввода/вывода. В ATtiny2313 их три группы: порт А, порт В и порт D. В группе порта А у нас три пина — это PA0, PA1,PA2. В группе порта В их уже восемь (PB0…PB7) и в группе порта D семь пинов (PD0…PD6). Итого в сумме 18 пинов, на каждый из которых можно навесить по светодиоду. То есть теоретически, наш МК может управлять 18 светодиодами, которые могут светится независимо друг от друга.
Характеристики светодиодов
Не забываем, что светодиоды делятся на индикаторные и осветительные. Индикаторные светодиоды обладают слабым свечением и используются для индикации каких-либо процессов, происходящих в электронной цепи. Для них характерно слабое свечение и малый ток потребления
Ну и осветительные светодиоды — это те, которые используются в ваших китайских фонариках, а также в LED -лампах
В нашей статье мы будет использовать индикаторные светодиоды. Светодиод — это токовый прибор, т.е. для его нормальной работы требуется номинальный ток, а не напряжение. При номинальном токе на светодиоде падает некоторое напряжение , которое зависит от типа светодиода (номинальной мощности, цвета, температуры). Ниже табличка, показывающая какое падение напряжения бывает на светодиодах разных цветов свечения при номинальном токе:
Поэтому подключать их напрямую к МК с питанием в 5 Вольт запрещено. Для них это обернется летальных исходом. В первую очередь сгорят те, кто требует для своей работы небольшого напряжения: красные, желтые и оранжевые светодиоды. Сгореть могут как светодиоды, так и выход МК. Чтобы такого не произошло, надо последовательно со светодиодом ставить токоограничивающий резистор. Лучше всего такой резистор подобрать с помощью магазина сопротивления , либо с помощью потенциометра, включенного по схеме реостата. После того, как мы добились умеренного свечения светодиода, замеряем наше сопротивление и ставим постоянный резистор близкий к измеренному номиналу.
Если такой возможности нет, то делаем простейший расчет.Например, у вас в арсенале имеются желтые светодиоды. Питание нашего МК 5 Вольт, следовательно, напряжение логической единички также 5 Вольт. Так как на желтом светодиоде у нас падение напряжения составляет примерно 2 Вольта, значит, на токо-ограничивающем резисторе должно упасть 3 Вольта. Ток через светодиод берем в среднем 15-20 мА. Исходя из закона Ома высчитываем значение сопротивления: I=U/R ——> R=U/I=3/15=0,2 килоома или 200 Ом. На практике, резистора, номиналом в 200-500 Ом, хватает за глаза.
Способы подключения светодиода на порт МК
1) Это прямой способ , при котором включение светодиода происходит при наличии высокого логического уровня, то есть единички на выходе порта МК:
2) Инверсный способ , при котором светодиод загорается только тогда, когда на выходе порта будет низкий логический уровень, то есть ноль.
Это самый простой расчет без всяких заморочек. Более подробно читайте в этой статье .
Для управления какой-либо мощной нагрузкой, можно применить схему с транзистором и реле:
О фьюзе RSTDISBL
Ножка PA2 может выполнять функцию не только вывода порта А, но и также участвует при программировании МК. Если с помощью фьюза «RSTDISBL» мы разрешим этому выводу «притвориться» выходом, то потом не сможем перепрограммировать данный МК с помощью ISP программатора, а другой программатор, я к сожалению, пока еще не приобрел. Но если вы собираетесь делать эту конструкцию навека, то можете без проблем поставить галочку на фьюз «RSTDISBL» и повесить на этот вывод светодиод). Поэтому, я не буду задействовать эту ножку МК как выход. То есть в сумме у меня будет 17 светодиодов, вместо 18. Жертвуя одни светодиодом, я даю себе право использовать многократно свой МК, не прибегая к помощи другого программатора.
Читайте сопротивление проводника .
Пишем программу для чайников
Далее открываем программу Atmel Studio 6 и начинаем писать программу. Эта программа специально написана для новичков, чтобы они уловили суть всего процесса.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 | < span style = «font-family: arial, helvetica, sans-serif;» > #include <avr/io.h> //подключаем библиотеку аппаратных описаний #define F_CPU 8000000UL //выставляем частоту МК #include <util/delay.h> //поключаем библиотеку задержек int main ( void ) { DDRB = 0b11111111 ; //выставляем все выводы порта B на выход PORTB = 0b00000000 ; // притягиваем все выводы порта B к нулю DDRD = 0b11111111 ; //выставляем все выводы порта D на выход PORTD = 0b00000000 ; //притягиваем все выводы порта D к нулю DDRA = 0b11111111 ; //выставляем все выводы порта А на выход PORTA = 0b00000000 ; // притягиваем все выводы порта А к нулю while ( 1 ) { PORTB = 0b00000001 ; // зажигаем 1-ый светодиод _delay_ms ( 50 ) ; // ждем 50 мсек PORTB = 0b00000000 ; // выключаем 1-ый светодиод _delay_ms ( 50 ) ; //ждем 50 мсек PORTB = 0b00000010 ; //зажигаем 2-ой светодиод _delay_ms ( 50 ) ; //задержка 50мсек PORTB = 0b00000000 ; //выключаем 2-ой светодиод _delay_ms ( 50 ) ; //задержка 50 мсек PORTB = 0b00000100 ; //зажигаем 3-ий светодиод _delay_ms ( 50 ) ; //задержка PORTB = 0b00000000 ; //выключаем 3-ий светодиод _delay_ms ( 50 ) ; //задержка PORTB = 0b00001000 ; //зажигаем 4-ый светодиод _delay_ms ( 50 ) ; //задержка PORTB = 0b00000000 ; //выключаем 4-ый светодиод _delay_ms ( 50 ) ; //задержка PORTB = 0b00010000 ; //зажигаем 5-ый светодиод _delay_ms ( 50 ) ; //задержка PORTB = 0b00000000 ; //выключаем 5-ый светодиод _delay_ms ( 50 ) ; //задержка PORTB = 0b00100000 ; //зажигаем 6-ой светодиод _delay_ms ( 50 ) ; //задержка PORTB = 0b00000000 ; //выключаем 6-ой светодиод _delay_ms ( 50 ) ; //задержка PORTB = 0b01000000 ; //зажигаем 7-ой светодиод _delay_ms ( 50 ) ; //задержка PORTB = 0b00000000 ; //выключаем 7-ой светодиод _delay_ms ( 50 ) ; //задержка PORTB = 0b10000000 ; //зажигаем 8-ой светодиод _delay_ms ( 50 ) ; //задержка PORTB = 0b00000000 ; //выключаем 8-ой светодиод _delay_ms ( 50 ) ; //задержка PORTA = 0b00000001 ; //зажигаем 1-ый светодиод порта А _delay_ms ( 50 ) ; //задержка PORTA = 0b00000000 ; //выключаем 1-ый светодиод порта А _delay_ms ( 50 ) ; //задержка PORTA = 0b00000010 ; //зажигаем 2-ой светодиод порта А _delay_ms ( 50 ) ; //задержка PORTA = 0b00000000 ; //выключаем 2-ой светодиод порта А _delay_ms ( 50 ) ; //задержка PORTD = 0b00000001 ; //включаем 1-ый светодиод порта D _delay_ms ( 50 ) ; //задержка PORTD = 0b00000000 ; //выключаем 1-ый светодиод порта D _delay_ms ( 50 ) ; //задержка PORTD = 0b00000010 ; //зажигаем 2-ой светодиод порта D _delay_ms ( 50 ) ; // задержка PORTD = 0b00000000 ; //выключаем 2-ой светодиод _delay_ms ( 50 ) ; //задержка PORTD = 0b00000100 ; // зажигаем 3-ий светодиод порта D _delay_ms ( 50 ) ; //задержка PORTD = 0b00000000 ; // выключаем 3-ий светодиод _delay_ms ( 50 ) ; //задержка PORTD = 0b00001000 ; //зажигаем 4-ый светодиод порта D _delay_ms ( 50 ) ; //задержка PORTD = 0b00000000 ; // выключаем 4-ый светодиод порта D _delay_ms ( 50 ) ; //задержка PORTD = 0b00010000 ; // зажигаем 5-ый светодиод порта D _delay_ms ( 50 ) ; //задержка PORTD = 0b00000000 ; //выключаем 5-ый светодиод порта D _delay_ms ( 50 ) ; //задержка PORTD = 0b00100000 ; //зажигаем 6-ой светодиод порта D _delay_ms ( 50 ) ; //задержка PORTD = 0b00000000 ; // гасим 6-ой светодиод порта D _delay_ms ( 50 ) ; // задержка PORTD = 0b01000000 ; // зажигаем 7-ой светодиод порта D _delay_ms ( 50 ) ; //задержка PORTD = 0b00000000 ; // гасим 7-ой светодиод порта D _delay_ms ( 50 ) ; //задержка } } < / span > |
Программа для профи-программистов
Для тех, кто уже не первый год изучает AVR, приведу оптимизированный код этой же самой программы:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 | < span style = «font-family: arial, helvetica, sans-serif;» > #include <avr/io.h> #define F_CPU 8000000UL #include <util/delay.h> #define time_delay 50 int main ( void ) { DDRB = 0xFF ; PORTB = 0x00 ; DDRA = 0xFF ; PORTA = 0x00 ; DDRD = 0xFF ; PORTD = 0x00 ; unsigned char i , j , k ; while ( 1 ) { for ( i = 0 ; i < = 7 ; i ++ ) { PORTB = ( 1 < < i ) ; _delay_ms ( time_delay ) ; } PORTB = 0x00 ; for ( j = 0 ; j < = 1 ; j ++ ) { PORTA = ( 1 < < j ) ; _delay_ms ( time_delay ) ; } PORTA = 0x00 ; for ( k = 0 ; k < = 6 ; k ++ ) { PORTD = ( 1 < < k ) ; _delay_ms ( time_delay ) ; } PORTD = 0x00 ; } } < / span > |
Разбор кода в программе для чайников
Так как я особо не силен в языке С, то разбор кода постараюсь объяснить своими словами.
Итак, первая строка:
1 | < span style = «font-family: arial, helvetica, sans-serif;» > #include <avr/io.h></span> |
это заголовочный файл, который подключает определения ввода/вывода для устройства в данном проекте
1 | < span style = «font-family: arial, helvetica, sans-serif;» > #define F_CPU 8000000UL</span> |
здесь мы указываем компилятору, с какой частотой будет работать наш МК. Как вы видите, я взял частоту в 8 Мегагерц.
1 | < span style = «font-family: arial, helvetica, sans-serif;» > #include <util/delay.h></span> |
подключаем библиотеку задержек, то есть это могут быть функции, как _delay_ms(), _delay_us()
1 2 3 | < span style = «font-family: arial, helvetica, sans-serif;» > int main ( void ) { . . . . . . . . . . . . . . . . . } < / span > |
тело программы
1 | < span style = «font-family: arial, helvetica, sans-serif;» > DDRB = 0b11111111 ; < / span > |
Все выводы порта В выставляем как выход А
1 | < span style = «font-family: arial, helvetica, sans-serif;» > PORTB = 0b00000000 ; < / span > |
Изначальное состояние всех выводов порта В равно логическому нулю
Аналогично устанавливаем порты А и D.
1 | < span style = «font-family: arial, helvetica, sans-serif;» > while ( 1 ) < / span > |
Зацикливаем тело программы в бесконечном цикле while
1 | < span style = «font-family: arial, helvetica, sans-serif;» > PORTB = 0b00000001 ; < / span > |
Здесь все просто, достаточно рассмотреть, как выглядят пины в МК с точки зрения самого МК. «0b» говорит о том, что мы записываем команду в двоичной системе счисления, что очень удобно в некоторых случаях для
наглядности процесса, особенно для новичков.
Итак, если в ячейке ноль, следовательно, этот вывод будет выдавать логический ноль. Если в ячейке единица, то и вывод будет выдавать логическую единичку. Значит, в нашей записи PORTB=0b00000001 у нас картина будет выглядеть следующим образом:
В этом случае у нас на PB0 будет логическая единица, то есть 5 Вольт. А если эти 5 Вольт подать на светодиод через токоограничивающий резистор, то светодиод загорится.
Пример для закрепления. Строка кода выглядит вот так:
1 | < span style = «font-family: arial, helvetica, sans-serif;» > PORTB = 0b00001111 ; < / span > |
Какие пины МК будут выдавать логическую единичку?
Вспоминаем нашу структуру и загоняем в нее наши биты:
Ответ: пины PB0, PB1, PB2 и PB3.
То есть в данном кусочке кода пины PB0, PB1, PB2 и PB3 одновременно выдадут логическую единичку, то бишь 5 Вольт.
Ну и задержка в миллисекундах:
1 | < span style = «font-family: arial, helvetica, sans-serif;» > _delay_ms ( 50 ) ; < / span > |
В данном случае это 50 миллисекунд
Эмуляция программы в Proteus и реальная работа гирлянды
Далее, нам надо создать HEX-файл. Как скомпилировать нашу программку и создать HEX-файл, читаем в этой статье.
После того, как у нас создался HEX-файл, мы его загружаем в наш виртуальный МК и запускаем процесс эмуляции в программе Proteus. Более подробно в этой статье.
Раз уж все так хорошо, почему бы не прошить реальный МК? О том, как прошивать МК AVR читайте в этой статье.
Зачем я вообще собрал эту схему бегущих огней? Чтобы ставить ее на новогоднюю елочку? Конечно же нет! Целью этой статьи была работа с портами ввода/вывода. В данной статье мы настраивали пины на выход для управления какой-нибудь нагрузкой. Вы без проблем можете поменять программу и решить для себя, как доработать эту «гирлянду». Меняя биты и задержки по времени, вы без труда сможете сделать различные эффекты и даже управлять какой-либо нагрузкой, подключив транзисторный ключ в паре с реле.
Кстати, такие готовые релейные модули вы без труда можете купить на Алике
Для промышленного оборудования стараются использовать программные логические контроллеры (ПЛК,PLC) которые идут в связке с модулями входов/выходов