
Рис. 7.13.Эквивалентная схема цепи протекания тока через замкнутую кнопку клавиатуры
Из приведенной на рис. 7.13 эквивалентной схемы видно, что для опроса колонки кнопок необходимо открыть нижний транзистор порта, подключенного к выбранной колонке. При этом нужно обеспечить запирание транзисторов, подключенных к остальным колонкам кнопок. Это позволит исключить неоднозначность определения номера нажатой кнопки. Для открывания транзистора достаточно записать в соответствующий бит порта логический 0, а для запирания - логическую 1. В результате коды опроса клавиатуры для схемы, приведенной на рис. 7.12, будут выглядеть следующим образом:
- код опроса первой колонки клавиатуры - 11110111b;
- код опроса второй колонки клавиатуры - 11111011b;
- код опроса третьей колонки клавиатуры - 11111101b;
- код опроса четвертой колонки клавиатуры - 11111110b.
То есть для опроса состояния кнопок клавиатуры потребуется выдача на порт Р1, по крайней мере, четырех кодов. При этом следует отметить, что скан-код, считанный с клавиатуры, содержит намного больше информации, чем просто номер замкнутого контакта. Например, при помощи скан-кода можно определить нажатие нескольких кнопок одновременно, поэтому лучше хранить во внутренней памяти контроллера непосредственно скан-код, а не готовый номер кнопки. Пример подпрограммы опроса клавиатуры приведен в листинге 7.23.


В результате работы этой подпрограммы нули появятся в битах, соответствующих цепям колонок и строк клавиатуры, к которым подключены нажатые кнопки. Использование операции побитового умножения & позволяет определить не только нажатие одиночной кнопки клавиатуры, но и одновременное нажатие двух кнопок, а также некоторых комбинаций трех кнопок клавиатуры.
Эта же подпрограмма может быть реализована и на языке программирования ассемблер. При этом она практически не будет отличаться от подпрограммы, приведенной выше (см. листинг 7.23). Пример реализации опроса клавиатуры на языке программирования ASM-51 приведен в листинге 7.24.


Интересной особенностью приведенной подпрограммы является использование команды sjmp, позволяющее превратить команду перехода по неравенству кодов cjne в команду перехода по равенству кодов. Остальные действия подробно пояснены комментариями и поэтому в дополнительных пояснениях не нуждаются.
Следующая подпрограмма обработки данных может из скан-кодов сформировать последовательности (строки) кодов нажатых кнопок, которые могут быть использованы как управляющие команды. Для хранения этих последовательностей можно использовать массив символов (в языке программирования ассемблер это просто соседние ячейки памяти).
При написании программы до сих пор считалось, что мы работаем с идеальными кнопками. Однако это не так. Обычно при нажатии и отпускании кнопки возникает переходный процесс, который называется дребезгом контактов. Начинающие разработчики аппаратуры применяют различные методы борьбы с этим явлением - от применения специальных схемотехнических решений до повторного опроса кнопок в течение некоторого времени.
Рассмотрим подробнее механизм возникновения явления дребезга контактов. Его причиной является упругость самих контактов. При нажатии на кнопку или при срабатывании какого-либо датчика контакты испытывают механическое импульсное воздействие. Обладая некоторой упругостью и массой, они начинают совершать колебательное движение: один контакт ударяется о другой (замыкание) и снова отходит (размыкание). Колебательный процесс продолжается некоторое время. При этом количество замыканий и размыканий контактов случайно и зависит от механических свойств контактов и механической силы, воздействующей на эти контакты. Обычно этот процесс длится от одной до восьми миллисекунд.
Если мы будем опрашивать состояние контактов с периодом, превышающим максимальную длительность дребезга, то даже не заметим, что они несколько раз замыкались или размыкались в промежутке между опросами контактов. Для этого можно включить в основной цикл программу, которая будет обеспечивать выполнение цикла один раз за строго определенное время.
Пример временной диаграммы сигнала на контактах кнопки приведен на рис. 7.14. На временной оси этого рисунка моменты считывания сигналов показаны рисками. Внизу рисунка обозначены номера временных слотов. На приведенной временной диаграмме четко просматривается зона дребезга контактов.
Для иллюстрации эффекта подавления дребезга контактов за счет ввода информации в строго определенные моменты времени на этой временной диаграмме выбран наихудший случай момента взятия отсчета. Этот момент совпадает с зоной дребезга контактов. При этом на выводе порта микроконтроллера может быть считан сигнал логического нуля или логической единицы. Но даже в этом случае дополнительный импульс в считанном сигнале не возникает! В случае считывания в момент дребезга контактов логического нуля сигнал, введенный в микроконтроллер, будет выглядеть как показано на рис. 7.14 сплошной линией, а в случае считывания логической единицы - пунктиром. Однако вне зависимости от того, какое значение сигнала мы считаем - нулевое или единичное, переход будет только один.

Рис. 7.14.Временные диаграммы напряжения на контактах кнопки и сигнала, введенного в микроконтроллер
Дребезг контактов приводит только к неопределенности обнаружения момента нажатия кнопки, которая не превышает периода опроса клавиатуры. Выберем это время. Так как мы уже определили, что время дребезга контактов не превышает 8 мс, то можно производить опрос портов, к которым подключены механические контакты, с периодом, который несколько больше, например, 10 мс. Это время и будет временем реакции системы. Но как обеспечить периодический опрос клавиатуры? Ведь программа в процессе выполнения может проходить по различным путям в зависимости от состояния опрашиваемых контактов и содержимого внутренних переменных микроконтроллера! В результате время выполнения программы (тела рабочего цикла) будет случайным. Для того чтобы время прохождения программы по циклу было строго фиксированным, можно воспользоваться таймером.
Таймер, являясь аппаратным элементом микроконтроллера, работает независимо от выполняемой программы, поэтому будет отсчитывать строго определенные промежутки времени. Если в конце выполнения цикла дожидаться срабатывания таймера, то время одного прохождения по циклу будет строго фиксированным. Единственное условие - максимальное время прохождения рабочего участка тела цикла должно быть меньше 10 мс. Но ведь то же самое требуется и с точки зрения максимального времени реакции системы. Если мы успеем несколько раз нажать на кнопки, а устройство не среагирует на эти нажатия, то что полезного можно сделать с его помощью? Если рабочая часть тела цикла будет выполнена быстро, то ждать срабатывания таймера придется долго. Если же проход по циклу будет выполняться долго, то после завершения рабочей части цикла долго ждать срабатывания таймера не придется. Соотношение времени выполнения рабочей части цикла и ожидания срабатывания таймера приведено на рис. 7.15.

Рис. 7.15. Соотношение времени выполнения рабочей части цикла и ожидания срабатывания таймера
Для организации периодического опроса клавиатуры требуется усложнить подпрограмму инициализации. Теперь в ней требуется настроить таймер на выбранный период времени, а значит задать и его режим работы. Выберем для задания равных промежутков времени таймер Т0. Он может обеспечить задание 10-мс промежутка времени только в режиме 16-разрядного таймера. Подпрограмма инициализации микроконтроллера, выполняющая настройку таймера, приведена в листинге. 7.25.