Лабораторная работа №3. Организация ввода/вывода информации. Механизм прерываний

Цель работы изучить основные приёмы, используемые при организации обмена данными через порты ввода/вывода (I/O).

Краткие теоретические сведения

Порты I/O относятся к периферии МК и являются средством его общения с внешними устройствами. Каждый порт имеет три регистра:

DDRx, PINx, PORTx. Где “x” имя порта (A,B,C,D и т.д.), количество которых различается, в зависимости от модели контроллера.

· DDRx определяет направление передачи данных порта на ввод либо вывод. Запись 0x00 настраивает порт на ввод данных. Например, запись в регистр DDRA значения 0b11111100, настроит нулевую и первую линии порта А на режим ввода, а остальные на вывод.

· PINx – регистр, в который читается состояние логических уровней на выводах контроллера.

· PORTx – регистр, из которого данные попадают на соответствующие выводы контроллера при выводе.

Вывод данных из контроллера производится командой OUT, а ввод –командой IN (например, in r16, PinA, или out PortB, r16). При этом соответствующие выводы предварительно должны быть настроены на ввод либо на вывод данных.

Для непосредственной работы с линиями портов ввода/вывода удобно пользоваться командами SBI, CBI и SBIS, SBIC. Первая устанавливает указанный бит в порту, а вторая сбрасывает. Например:

cbi PortA,7 ; обнулить 7й бит в регистре ввода-вывода PortA

sbi PortB,6 ; выставить 6й бит в регистре ввода-вывода PortB

SBIC проверяет состояние разряда регистра ввода/вывода. Если разряд сброшен, команда, следующая за sbic Address,bit пропускается. SBIS наоборот, пропускает команду, следующую за sbis Address,bit если разряд установлен. Следует помнить, что эти команды актуальны только в пределах пространства регистров I/O.

Задание на лабораторную работу

Задача 3.1. Написать программу, обеспечивающую отображение состояния i-го и j-го битов операнда на линиях порта B i и j соответственно. Адрес операнда 02** в памяти данных. Собрать модель схемы в Proteus и подключить к выводам порта B i и j светодиоды, отображающие состояние операнда. Программу проверить для операндов FF, 0F, F0, 00.

Задача 3.2. Написать программу, обеспечивающую отображение состояния клавиши, подключенной к линии i порта B, с помощью светодиода, подключенного к линии j порта D (нажато – горит, отжато – не горит). Промоделировать работу схемы в Proteus.

Задача 3.3. Написать программу, обеспечивающую подсчёт нажатий на клавишу, подключенную к линии i порта B. Отобразить десятичный эквивалент количества нажатий на двух семисегментных индикаторах, подключенных к порту D. Сброс в 0 клавишей на линии j порта B. При достижении значения счётчика 99 значение больше не увеличивать. Промоделировать работу схемы в Proteus.

Задача 3.4. Написать программу, обеспечивающую увеличение либо уменьшение десятичного счётчика в зависимости от нажатия клавиш “+” или “-“, подключенных к линиям i и j порта D. Отобразить десятичное значение на двух семисегментных индикаторах, подключенных к порту B. При опросе клавиш задействовать механизм внешнего прерывания. Сброс в 0 клавишей на линии k порта D. При достижении значения счётчика 99 значение больше не увеличивать, а также не уменьшать менее 0. Промоделировать работу схемы в Proteus.

Задача 3.5. Написать программу, обеспечивающую работу энкодера, подключенного к выводам порта D i и j. Отобразить десятичный эквивалент поворота ручки энкодера на двух семисегментных индикаторах, подключенных к порту B. При опросе энкодера задействовать механизм внешнего прерывания. Сброс в 0 клавишей на линии k порта D. При достижении значения счётчика 99 значение больше не увеличивать, а также не уменьшать менее 0. Промоделировать работу схемы в Proteus.

Примеры решения задач

Задача 3.1. Составим программу, обеспечивающую отображение состояния нулевого бита операнда на нулевой линии порта B. Собрать модель схемы в Proteus и подключить к нулевому выводу порта B светодиод, отображающий состояние нулевого бита операнда. Программу проверить для операндов FF и 00.

Соответствующая программа имеет вид:

.include "m2560def.inc"

init: ldi r16, 0x01 ; настройка на вывод линии 0 порта B

out DDRB, r16

ldi r16, 0xFF ; операнд

ldi r17, 0x01 ; маска для проверки на 1 бита 0

main: and r16, r17 ; выделить нулевой бит

brne on ; переход по ненулю

ldi r16, 0x00 ; вывод на линию 0 порта B "0"

out PORTB, r16

rjmp end ; обход включения линии порта

on: ldi r16, 0x01 ; вывод на линию 0 порта B "1"

out PORTB, r16

end: rjmp end ; остановка программы

Модель микроконтроллерной системы Arduino для разработанной программы в среде Proteus выглядит следующим образом:

Лабораторная работа №3. Организация ввода/вывода информации. Механизм прерываний - student2.ru

Светодиод подключен катодом к минусовому проводу ”земли“, поэтому включается подачей логической ”1“ с порта B. Резистор необходим для согласования уровня напряжения на выходе Arduino (5 вольт) с напряжением включения светодиода (3 вольта).

Эмуляция работы устройства показывает, что при значении операнда FF светодиод включен, а при значении 00 – выключен.

Задача 3.2. Составим программу, обеспечивающую отображение состояния клавиши, подключенной к линии 0 порта B, с помощью светодиода, подключенного к линии 0 порта D (нажато – горит, отжато – не горит). Промоделировать работу схемы в Proteus.

Для выполнения действий во время работы устройства алгоритм программы примет вид бесконечного цикла, который постоянно опрашивает состояние клавиши на входе, в соответствии с этим производит вычисления, и выдаёт управляющее воздействие на выход.

При опросе кнопки необходимо учитывать, что она работает в инверсной логике, т.е. её нажатие вызывает появление логического нуля в соответствующем разряде порта B. Соответствующая программа имеет вид:

.include "m2560def.inc"

init: ldi r16, 0b00000000 ; настройка на ввод линии 0 порта B

out DDRB, r16

ldi r16, 0b00000001 ; настройка на вывод линии 0 порта D

out DDRD, r16

ldi r19, 0b00000001 ; маска для инверсии 0 бита

main: in r16, PINB

eor r16, r19 ; инвертирование 0 бита

out PORTD,r16 ; вывод на линию 0 состояния бита 0 r16

rjmp main ; организация бесконечного цикла

Модель микроконтроллерной системы Arduino для разработанной программы в среде Proteus выглядит следующим образом:

Лабораторная работа №3. Организация ввода/вывода информации. Механизм прерываний - student2.ru

При использовании такой схемы на практике необходимо учитывать эффект дребезга контактов кнопки, которая выражается в многократном замыкании-размыкании цепи в момент соединения контактов. Это воспринимается как быстрая смена логических уровней сигнала, или как многократное нажатие клавиши. Поэтому кнопку следует считать нажатой, если на входе МК в течении более чем 0,01с будет присутствовать низкий уровень.

Задача 3.3. Составим программу, обеспечивающую подсчёт нажатий на клавишу, подключенную к линии 0 порта B. Отобразить десятичный эквивалент количества нажатий на семисегментном светодиодном индикаторе, подключенному к четырём младшим линиям порта D. Сброс в 0 клавишей на линии 1 порта B. При достижении значения счётчика 9 значение больше не увеличивать. Промоделировать работу схемы в Proteus.

.include "m2560def.inc"

init: ldi r16, 0b00000000 ; настройка на ввод линий 0,1 порта B

out DDRB, r16

ldi r16, 0b00001111 ; настройка на вывод линий 0-3 порта D

out DDRD, r16

clr r17 ; сброс счётчика

bclr 6 ; разрешить увеличение счёта b6SREG=0

main: sbis PINB, 1 ; проверить кнопку RESET

clr r17 ; если нажата, то сброс счётчика

sbic PINB, 0 ; проверить кнопку +

rjmp off ; переход если не нажато

brbs 6, nozero ; можно увеличить счёт?

inc r17

nozero:bset 6 ; запретить увеличение счёта b6SREG=1

rjmp end ; обход разрешения

off: bclr 6 ; разрешить увеличение счёта

end: cpi r17, 0x0A ; сравнить r17 с числом A

brne indic ; перейти, если r16 /= A

dec r17

indic: out PORTD,r17 ; вывод

rjmp main ; организация бесконечного цикла

Для управления семисегментным индикатором целесообразно применить дешифратор, для преобразования двоичного кода, формируемого контроллером, в управляющие сигналы, включающие отдельные светодиоды, которые соответствуют цифрам кода. В качестве такого элемента используем микросхему 74LS47.

Модель микроконтроллерной системы Arduino для разработанной программы в среде Proteus выглядит следующим образом:

Лабораторная работа №3. Организация ввода/вывода информации. Механизм прерываний - student2.ru

Задача 3.4. Составим программу, увеличивающую либо уменьшающую десятичный счётчик по нажатию клавиш “+” или “-“. Подключим их к линиям 0 и 1 порта D. Отобразим десятичное значение на семисегментном индикаторе, подключенном к порту B. При опросе клавиш используем внешнее прерывание. Сброс в 0 клавишей на линии 2 порта D. Промоделируем работу схемы в Proteus.

Для работы с прерываниями необходимо описать таблицу векторов для данного типа МК через директивы ассемблера .org. Из документации на МК ATmega2560 следует, что внешние прерывания INTx имеют диапазон адресов $0002.. $0010. При этом INT0 принимается с линии порта PD0 микроконтроллера, а INT1 – c PD1 соответственно. Для желаемых INTx по заданным адресам записывается команда перехода на обработчик rjmp Ext_INT. Подпрограмма-обработчик должна быть как можно короче и обязательно заканчиваться командой reti. Основная программа начинается с метки Reset, где настраивается указатель стека и внешние прерывания разрешаются локально (регистр EIMSK), а также условие срабатывания по спаду входного сигнала в регистре EICRA (адрес в пространстве ввода/вывода 0x69).

.include "m2560def.inc"

.cseg

.org $0000 rjmp Reset ; (Reset)

.org $0002 rjmp Ext_INT0 ; (INT0) External Interrupt Request 0

.org $0004 rjmp Ext_INT1 ; (INT1) External Interrupt Request 1

.org $0006 reti ; (INT2) External Interrupt Request 2

.org $0008 reti ; (INT3) External Interrupt Request 3

.org $000A reti ; (INT4) External Interrupt Request 4

.org $000C reti ; (INT5) External Interrupt Request 5

.org $000E reti ; (INT6) External Interrupt Request 6

.org $0010 reti ; (INT7) External Interrupt Request 7

.org $0012 reti ; (PCINT0) Pin Change Interrupt Request 0

.org $0014 reti ; (PCINT1) Pin Change Interrupt Request 1

.org $0016 reti ; (PCINT2) Pin Change Interrupt Request 2

.org $0018 reti ; (WDT) Watchdog Time-out Interrupt

.org $001A reti ; (TIMER2_COMPA) Timer/Counter2 Compare Match A

.org $001C reti ; (TIMER2_COMPB) Timer/Counter2 Compare Match B

.org $001E reti ; (TIMER2_OVF) Timer/Counter2 Overflow

.org $0020 reti ; (TIMER1_CAPT) Timer/Counter1 Capture Event

.org $0022 reti ; (TIMER1_COMPA) Timer/Counter1 Compare Match A

.org $0024 reti ; (TIMER1_COMPB) Timer/Counter1 Compare Match B

.org $0026 reti ; (TIMER1_COMPC) Timer/Counter1 Compare Match C

.org $0028 reti ; (TIMER1_OVF) Timer/Counter1 Overflow

.org $002A reti ; (TIMER0_COMPA) Timer/Counter0 Compare Match A

.org $002C reti ; (TIMER0_COMPB) Timer/Counter0 Compare Match B

.org $002E reti ; (TIMER0_OVF) Timer/Counter0 Overflow

.org $0030 reti ; (SPI_STC) Serial Transfer Complete

.org $0032 reti ; (USART0_RX) USART0 Rx Complete

.org $0034 reti ; (USART0_UDRE) USART0 Data Register Empty

.org $0036 reti ; (USART0_TX) USART0 Tx Complete

.org $0038 reti ; (ANALOG_COMP) Analog Comparator

.org $003A reti ; (ADC) ADC Conversion Complete

.org $003C reti ; (EE_READY) EEPROM Ready

.org $003E reti ; (TIMER3_CAPT) Timer/Counter3 Capture Event

.org $0040 reti ; (TIMER3_COMPA) Timer/Counter3 Compare Match A

.org $0042 reti ; (TIMER3_COMPB) Timer/Counter3 Compare Match B

.org $0044 reti ; (TIMER3_COMPC) Timer/Counter3 Compare Match C

.org $0046 reti ; (TIMER3_OVF) Timer/Counter3 Overflow

.org $0048 reti ; (USART1_RX) USART1 Rx Complete

.org $004A reti ; (USART1_UDRE) USART1 Data Register Empty

.org $004C reti ; (USART1_TX) USART1 Tx Complete

.org $004E reti ; (TWI) 2-wire Serial Interface

.org $0050 reti ; (SPM_RDY) Store Program Memory Ready

.org $0052 reti ; (TIMER4_CAPT) Timer/Counter4 Capture Event

.org $0054 reti ; (TIMER4_COMPA) Timer/Counter4 Compare Match A

.org $0056 reti ; (TIMER4_COMPB) Timer/Counter4 Compare Match B

.org $0058 reti ; (TIMER4_COMPC) Timer/Counter4 Compare Match C

.org $005A reti ; (TIMER4_OVF) Timer/Counter4 Overflow

.org $005C reti ; (TIMER5_CAPT) Timer/Counter5 Capture Event

.org $005E reti ; (TIMER5_COMPA) Timer/Counter5 Compare Match A

.org $0060 reti ; (TIMER5_COMPB) Timer/Counter5 Compare Match B

.org $0062 reti ; (TIMER5_COMPC) Timer/Counter5 Compare Match C

.org $0064 reti ; (TIMER5_OVF) Timer/Counter5 Overflow

.org $0066 reti ; (USART2_RX) USART2 Rx Complete

.org $0068 reti ; (USART2_UDRE) USART2 Data Register Empty

.org $006A reti ; (USART2_TX) USART2 Tx Complete

.org $006C reti ; (USART3_RX) USART3 Rx Complete

.org $006E reti ; (USART3_UDRE) USART3 Data Register Empty

.org $0070 reti ; (USART3_TX) USART3 Tx Complete

.org INT_VECTORS_SIZE ; Конец таблицы прерываний

Ext_INT0: ; обработчик прерывания по входу INT0

inc r17 ; счётчик++

cpi r17, 0x0A ; сравнить r17 с числом A

brne ret0 ; перейти, если r17 /= A

dec r17

ret0: reti ; выход из обработчика прерывания

Ext_INT1: ; обработчик прерывания по входу INT1

dec r17 ; счётчик--

cpi r17, 0xFF ; сравнить r17 с числом FF

brne ret1 ; перейти, если r17 /= FF

inc r17 ; счётчик++

ret1: reti ; выход из обработчика прерывания

Reset: ldi r16,Low(RAMEND) ; Старт программы

out SPL,r16 ; Обязательная инициализация стека

ldi r16,High(RAMEND); Указатель стека устанавливается на конец ОЗУ

out SPH,r16

ldi r16,0b00000011 ; Разрешаем прерывания INT0 и INT1 локально

out EIMSK,r16

ldi r16,0b00001010 ; Настройка условия генерации прерывания

sts EICRA,r16 ; по спаду входного сигнала

sei ; Разрешаем прерывания глобально

ldi r16,0b00000000 ; настройка на ввод линий 0,1,2 порта D

out DDRD,r16

ldi r16,0b00001111 ; настройка на вывод линий 0-3 порта B

out DDRB,r16

clr r17 ; сброс счётчика

main: sbis PIND,2 ; проверить кнопку RESET

clr r17 ; если нажата, то сброс счётчика

out PORTB,r17 ; вывод

rjmp main

Модель микроконтроллерной системы Arduino для разработанной программы в среде Proteus выглядит следующим образом:

Лабораторная работа №3. Организация ввода/вывода информации. Механизм прерываний - student2.ru

Задача 3.5. Составим программу, увеличивающую либо уменьшающую десятичный счётчик при воздействии на энкодер, подключенный к линиям 0 и 1 порта D. Отобразим десятичное значение на семисегментном индикаторе, подключенном к порту B. При опросе энкодер задействуем механизм внешнего прерывания. Сброс в 0 клавишей на линии 2 порта D. Промоделируем работу схемы в Proteus.

Энкодер – это датчик положения движущихся частей, который имеет две контактные пары, формирующие при вращении его ротора две импульсные последовательности, сдвинутые на 90°. По этому сдвигу можно судить о направлении вращения энкодера.

Лабораторная работа №3. Организация ввода/вывода информации. Механизм прерываний - student2.ru

На диаграмме показана зависимость выходов А и В друг от друга при вращении энкодера по часовой или против часовой стрелки.

Всякий раз, когда по прерыванию INT0 сигнал А переходит от нуля к положительному уровню, необходимо считывать значение выхода В. Если B в этот момент находится в положительном состоянии, значит энкодер вращается по часовой стрелке, иначе – против часовой стрелки. В зависимости от полученного результата увеличивается или уменьшается значение счётчика.

.include "m2560def.inc"

.cseg

.org $0000 rjmp Reset ; (Reset)

.org $0002 rjmp Ext_INT0 ; (INT0) External Interrupt Request 0

.org INT_VECTORS_SIZE ; Конец таблицы прерываний

Ext_INT0: ; обработчик прерывания по входу INT0

inc r17 ; счётчик++

cpi r17, 0x0A ; сравнить r17 с числом A

brne ret0 ; перейти, если r17 /= A

dec r17

ret0: reti ; выход из обработчика прерывания

Наши рекомендации