• Добро пожаловать на компьютерный форум Tehnari.ru. Здесь разбираемся с проблемами ПК и ноутбуков: Windows, драйверы, «железо», сборка и апгрейд, софт и безопасность. Форум работает много лет, сейчас он переехал на новый движок, но старые темы и аккаунты мы постарались сохранить максимально аккуратно.

    Форум не связан с магазинами и сервисами – мы ничего не продаём и не даём «рекламу под видом совета». Отвечают обычные участники и модераторы, которые следят за порядком и качеством подсказок.

    Если вы у нас впервые, загляните на страницу о форуме и правила – там коротко описано, как задать вопрос так, чтобы быстро получить ответ. Чтобы создавать темы и писать сообщения, сначала зарегистрируйтесь, а затем войдите под своим логином.

    Не знаете, с чего начать? Создайте тему с описанием проблемы – подскажем и при необходимости перенесём её в подходящий раздел.
    Задать вопрос Новые сообщения Как правильно спросить
    Если пришли по старой ссылке со старого Tehnari.ru – вы на нужном месте, просто продолжайте обсуждение.

Проблема с опросом энкодера на внешних прерываниях

Регистрация
26 Мар 2016
Сообщения
16
Реакции
1
Баллы
0
Проблема с опросом энкодера на внешних прерываниях

Такая тема уже была кажется, но с такой проблемой вроде никто не сталкивался. Задание следующее:Отобразить номер своего варианта на ЖКИ индикатор с возможностью
изменения значения энкодером в большую (до 999) или меньшую сторону.
Опрос энкодера реализовать на внешних прерываниях.
Вот код:
Посмотреть вложение KOD_2.txt
Алгоритм стандартный, но при повороте ручки энкодера влево-вправо он либо в обе стороны прибавляет, либо в обе стороны отнимает. В чем может быть дело?
 
Скажите, а зачем там вообще таймер?
Вы разрешили прерывание по спаду INT0, объявили обработчик - что еще надо? Флаг прерывания сам сбросится при входе в обработчик.

Или же у вас дребезг контактов?

Но тогда лучше вообще все по таймеру, а внешнее прерывание запретить.
Читать по таймеру, сравнивать с предыдущим.
 
Недавно боролся с похожей проблемой. Поборол. Суть в следующем.При инициализации:1. Привязываем прерывание к срезу импульса на выходе А энкодера.Обработчик прерываний:1. Запрещает прерывания.2. Взводит глобальный флаг.Основной код программы:1. Если взведен флаг, выполняем последующий код, иначе - пропускаем.2. Задержка 3 мс для завершения возможного переходного процесса.3. Читаем выход А энкодера. Если высокий - прерывание ложное, сбрасываем глобальный флаг, разрешаем прерывания, пропуск последующего кода.4. Читаем выход В энкодера - если высокий, добавляем нужную величину, низкий - убавляем и т.д., вобщем стандартный вариант. Перед завершением обработки сбрасываем глобальный флаг и разрешаем прерывания

Выше предложенный алгоритм заработал, однако программа работает не оптимально: иногда при повороте энкодера возникает осечка, либо прибавляет/отнимает 2, вместо 1.
Пробовал перебрать частоту тиков таймера, ничего не помогает.
Код:
#include "hd44780.h"
#include <avr/io.h>
#include <avr/interrupt.h>
#include <util/delay.h>
#define ENC_PORT PORTD
#define ENC_PIN PIND
#define A_PIN 2
#define B_PIN 1
#define Comapare_Mask (1<<A_PIN)|(1<<B_PIN)
volatile uint8_t flag=0;
void timer0_on()
{
TCNT0=0x00; //записываем 0 в счетный регистр
TCCR0|=(1<<CS02)|(0<<CS01)|(1<<CS00);//устанавливаем частоту тиков
TIMSK|=(0<<OCIE0)|(1<<TOIE0);//устанавливаем прерывания по переполнению
}
void timer0_off()
{
TCCR0|=(0<<CS02)|(0<<CS01)|(0<<CS00);
}
int main(void)
{ int count=14;
lcd_init();
lcd_clear();
lcd_out4(count,2);
MCUCR |= (1<<ISC01)|(0<<ISC00);//прерывание по спаду
GICR = (1<<INT0); //разрешение прерываний
ENC_PORT |= Comapare_Mask;//опрос энкодера
sei();
while(1)
{
  if (flag==1)
  {
   timer0_on();
   if ((ENC_PIN&(1<<A_PIN)) >>A_PIN==1) //считываем вывод А
   {
        flag = 0;//если уровень высокий сбрасываем флаг
   }
   else
   {
        if ((ENC_PIN&(1<<B_PIN)) >>B_PIN==1)
        {
         count--;
         if (count<0) count=9999;
        }
        else
        {
         count++;
         if (count>9999)
          count=0;
        }
        flag = 0;
   }
  }
  lcd_out4(count,2);
};
}
ISR (INT0_vect)
{
GICR = (0<<INT0); //запрещаем прерывания
flag = 1;//взводим глобальный флаг
}
ISR (TIMER0_OVF_vect)
{
timer0_off();
GICR = (1<<INT0);
}
 
Назад
Сверху