Встраиваемые системы. Проектирование приложений на микроконтроллерах семейства 68HC12/HCS12 с применением языка С — страница 41 из 70

.

.

unsigned char INМASK = 0х00;

unsigned char OUTМASK = 0xff;

unsigned char PORTA_value;

DDRA = INМASK; /*порт PORTA на ввод*/

DDRB = OUTМASK; /*порт PORTB на вывод*/

PORTA_value = PORTA; /*cчитать портPORTA*/

PORTB = PORTA_value; /*записать в порт PORTB*/

.

.

5.5. Подавление механического дребезга контактов переключателей

Ранее, в параграфе 5.2, Вы познакомились с различными типами механических переключателей. Переключатели обладают двумя устойчивыми состояниями, в которых замыкаются или размыкаются электрические контакты. Включенный по схеме рис. 5.4,а переключатель обеспечивает высокий логический уровень сигнала на входе МК, если пара его контактов разомкнута. В нажатом состоянии контакты замыкаются, и на входе МК формируется напряжение низкого логического уровня.

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

Однако реальные переключатели ведут себя иначе. Так при нажатии клавиши рис. 5.4,а контакты многократно замыкаются и размыкаются, пока не окажутся в установившемся замкнутом состоянии. Это явление называется механическим дребезгом переключателя. Микроконтроллеры работают на частотах в несколько МГц, что остаточно для того, чтобы зафиксировать многократное изменение состояния переключателя во время дребезга. Однако реакция программы управления на каждое переключение во время дребезга было бы ошибкой. Поэтому следует применять специальные методы подавления эффекта механического дребезга. Различают аппаратные и программные способы подавления дребезга контактов. Мы рассмотрим несколько вариантов для каждого из этих способов.

5.5.1. Аппаратная защита от механического дребезга контактов

На рис. 5.10 представлена схема, которая осуществляет защиту от дребезга механических переключателей [Horowitz и Hill, 1989]. Основным элементом этой схемы является триггер Шмитта (74HC14).

Триггер Шмита отличается от других логических элементов, во-первых, тем, что на его входы можно подавать аналоговые сигналы. И если для обычного логического элемента время изменения входного сигнала из состояния 0 в состояние 1 должно составлять всего несколько наносекунд, то для триггера Шмитта это время может быть любым, в том числе несколько десятков или сотен миллисекунд, которые потребуются нам для устранения дребезга контактов. Во-вторых, передаточная характеристика триггера Шмитта обладает гистерезисом: уровень входного напряжения, при котором выход триггера переключается из 0 в 1, превышает уровень напряжения переключения из 1 в 0 примерно на 0,5…0,8 В. Эффект гистерезиса позволяет подавить звон входного сигнала. Незначительные по амплитуде высокочастотные колебания, наложенные на монотонно изменяющуюся постоянную составляющую входного сигнала, не будут приводить к многократному изменению выходного сигнала, поскольку абсолютная величина входного сигнала окажется внутри петли гистерезиса.

Рассмотрим работу схемы подавления дребезга контактов переключателя. При разомкнутых контактах напряжение конденсатора равно VCC, на выходе триггера Шмита формируется низкий логический уровень, поскольку все триггеры Шмитта в интегральном исполнении инвертируют входной сигнал. Если клавишу только что нажали, то конденсатор начинает разряжаться через резистор и замкнутые контакты переключателя. Время его разряда определяется постоянной времени RC. Для приведенных на рис. 5.10,a номиналов постоянная времени равна 47 мс. Напряжение на конденсаторе будет убывать немонотонно, поскольку на интервале дребезга контакты то замкнутся, то разомкнутся. В соответствии с их положением конденсатор то разряжается, то заряжается. Но большая постоянная времени цепей разряда и заряда не позволяет напряжению на конденсаторе измениться во время дребезга столь сильно, чтобы произошло переключение триггера Шмитта. И лишь когда дребезг закончится и произойдет длительное замыкание контактов переключателя, конденсатор разрядится до нуля, и на выходе триггера Шмитта установится высокий логический уровень. При размыкании контактов переключателя процесс будет происходить в обратном порядке. Эффект подавления дребезга в рассмотренной схеме будет наблюдаться только тогда, когда постоянная времени цепей разряда и заряда конденсатора будет сравнима с длительностью самого эффекта механического дребезга. Для маломощных переключателей его длительность измеряется единицами и десятками мс, поэтому в нашем примере номиналы резисторов и конденсатора выбраны правильно.

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

5.5.2. Программная защита от механического дребезга контактов

Существует несколько методов программной защиты от эффекта дребезга механических контактов. Мы кратко остановимся на двух наиболее часто используемых:

• Прикладная программа проводит мониторинг состояния контакта, опрашивая линию порта и контролируя уровень логического сигнала. Как только программа зафиксировала изменение сигнала, она фиксирует это изменение и формирует задержку 100…200 мс. За это время дребезг закончится, и контакт зафиксируется в установившемся состоянии. Если это новое состояние совпадет с первым зафиксированным изменением, то произошло переключение контакта. В противном случае имела место помеха.

• Прикладная программа также производит мониторинг контакта. После обнаружения первого переключения программа продолжает постоянно опрашивать контакт в течение тех же 100…200 мс. Если к концу интервала опроса состояние не стабилизировалось и постоянно считывается то 0 то 1, то это помехи, а если установилось в одном из состояний, то произошло переключение.

а) При нажатии клавиши на выходе формируется высокий логический уровень

б) При нажатии клавиши на выходе формируется низкий логический уровень

Рис. 5.10. Примеры схем подавления механического дребезга контактов 

5.5.3. Пример программной защиты

Для того чтобы продемонстрировать пример реального программного кода для ввода и вывода дискретных управляющих воздействий, предположим, что восемь DIP переключателей подключены к порту PORTB микроконтроллера, а состояние выходов порта PORTC индицируются с использованием схемы двухцветного индикатора. Общая структура микропроцессорной системы нашего примера представлена на рис. 5.11. В примере мы используем аппаратную защиту и программную защиту от дребезга контактов. Программная защита реализуется по второму способу.

Рис. 5.11. Функциональная схема аппаратных средств для примера 5.3.3


/*----------------------------------------------------------------------*/

/*filename: debounced_swith.c                                           */

/*осуществляет опрос 8 переключателей с программной защитой от дребезга */

/*выполняет специальные действия для каждого нажатого переключателя     */

/*зажигает зеленый светодиод в разряде активизированного переключателя  */

/*----------------------------------------------------------------------*/

/*подключаемые файлы*/

#include <912b32.h>


/*используемые функции*/

int process_valid_input(unsigned char input_value); //управление

                                                    //процессом опроса в реальном времени

void initialize_ports(void); //инициализация портов

void timer_init(void); //инициализация таймера


//глобальные переменные

int keep_going=1; //переменная цикла

unsigned char old_PORTB= 0xff; //предыдущее значение порта PORTB

unsigned char new_PORTB; //новое значение порта PORTB


void main{void) {

 initialize_ports();

 timer_init();

 while(keep_going) {

  new_PORTB=PORTB; //читать порт PORTB

  if (new_PORTB != old_PORTB) {

   //выполнять, если значение порта изменилось

   swith(new_PORTB) {

   case 0xFE: //переключатель PB0

    if (process_valid_input(new_PORTB)) //процедура

                                        //антидребезга

    { //выполнять действия, связанные

     : //с нажатием клавиши PB0

     :

     PORTC = 0х01;//зажечь зеленый в разряде PB0

     keep_going = 1;

    }

    break;

   case 0xFD: // переключатель PB01

    if (process_valid_input(new_PORTB)) // процедура

                                        // антидребезга

    { //выполнять действия, связанные

     : //с нажатием клавиши PB1

     PORTC = 0х02; //зажечь зеленый в разряде PB1

     keep_going = 1;

    }

    break;

   case 0x7F: // переключатель PB7

    if (process_valid_input(new_PORTB)) // процедура

         //антидребезга

    { //выполнять действия, связанные

     : //с нажатием клавиши PB7

     PORTC = 0х80; //зажечь зеленый в разряде PB7

     keep_going = 1;

    }

    break;

   default:

    ; //аll other cases

   } //окончание распозвания, какая клавиша нажата (switch(new_PORTB))

  } //окончание обработки нажатой клавиши (if new_PORTB)

  old_PORTB = new_PORTB; //обновить PORTB

 } //окончание (while(keep_going))

} //окончание (main)


/*-----------------------------------------------------------------*/

/* Функция void initialize_ports устанавливает режим работы портов */

/*-----------------------------------------------------------------*/

void initialize_ports(void) {

 DDRC=0xff; //порт PORTC на вывод

 PORTC=0x00; //выходы порта PORTC в 0, зажгутся красные светодиоды

 DDRB=0x00; //порт PORTB на ввод

}


/*--------------------------------------------------------------------*/

/* process_valid_input: проверка стабильности PORTB в течение периода */

/* переполнения таймера                                               */

/*--------------------------------------------------------------------*/

int process_valid_input(unsigned char portx) {

 int valid_input; //флаг результата опроса порта

 int int_value; //время входа в процедуру

 valid_input = TRUE; //установить флаг

 int_value = TCNT; //читать текущий момент времени

 while (int_value != TCNT) //повторять пока значения времени снова не

                           //сравняются

 {

  if (portx==PORTB) //порт PORTB остается стабильным

   valid_input = TRUE;

  else //порт PORTB изменил состояние

   valid_input = FALSE;

  if (!valid_input) //если порт изменил состояние, то выйти

                    //из цикла

   break;

 }

 return valid_input;

}


/*------------------------------------------------------------------------*/

/* Функция timer_init инициализирует таймер. Частота системной шины равна */

/*8*МГц                                                                   */

/*------------------------------------------------------------------------*/

void timer_init(void) {

 TMSK1 = 0х00; //запретить прерывания

 TMSK2 = 0х02; //частота тактировании 2 МГц

 TSCR = 0х80; //разрешить работу модуля таймера

}

/*-------------------------------------------------------------------------*/

5.6. Жидкокристаллические индикаторы