реклама на сайте
подробности

 
 
 
Reply to this topicStart new topic
> динамическая индикация + клавиши, помогите
Lost_Viking
сообщение Apr 25 2008, 17:15
Сообщение #1


Частый гость
**

Группа: Участник
Сообщений: 168
Регистрация: 25-04-08
Пользователь №: 37 091



Не пойму в чем дело...

Задумка такая: есть 4 кнопки и 3-х разрядный 7-и сегментный индикатор. Кнопки вешаю на PB0..3, индикатор на PC0...7 (a...h) и на PD0..2 (три разряда). Каждая кнопка отвечает за прибавление единицы к разряду, после девятки возвращаемся на нуль. четвертая кнопка отвечает за сброс значения.

Цитата
/*****************************************************
This program was produced by the
CodeWizardAVR V1.25.9 Evaluation
Automatic Program Generator
© Copyright 1998-2008 Pavel Haiduc, HP InfoTech s.r.l.
http://www.hpinfotech.com

Project :
Version :
Date : 25.04.2008
Author : Freeware, for evaluation and non-commercial use only
Company :
Comments:


Chip type : ATmega16
Program type : Application
Clock frequency : 1,000000 MHz
Memory model : Small
External SRAM size : 0
Data Stack size : 256
*****************************************************/

#include <mega16.h>
#include <DELAY.H>

int res=0;
int e=0;
int d=0;
int s=0;

int inp (void) {
if (PINB==0xFE) e++; //прибавляем единицы
if (e==10) e=0; // если единицы достигли 10, то нуль.
if (PINB==0xFD) d++;
// if (d==10) d=0;
if (PINB==0xFB) s++;
// if (s==10) s=0;
if (PINB==0xF7) { // сброс
e=0;
d=0;
s=0; };
res=e+d*10+s*100; // само получившееся число
return res;
}





int segment_decoder(int num) //эту функцию я своровал с одной ветки форума smile.gif)
{
int result = 0x3F;
switch(num)
{
case 0: result = 0x3F;
break;
case 1: result = 0x06;
break;
case 2: result = 0x5B;
break;
case 3: result = 0x4F;
break;
case 4: result = 0x66;
break;
case 5: result = 0x6D;
break;
case 6: result = 0x7D;
break;
case 7: result = 0x07;
break;
case 8: result = 0x7F;
break;
case 9: result = 0x6F;
break;
default: result = 0x3F;
break;
}
return result;
}
// Declare your global variables here

void main(void)
{
// Declare your local variables here

// Input/Output Ports initialization
// Port A initialization
// Func7=In Func6=In Func5=In Func4=In Func3=In Func2=In Func1=In Func0=In
// State7=T State6=T State5=T State4=T State3=T State2=T State1=T State0=T
PORTA=0x00;
DDRA=0x00;

// Port B initialization
// Func7=In Func6=In Func5=In Func4=In Func3=In Func2=In Func1=In Func0=In
// State7=T State6=T State5=T State4=T State3=P State2=P State1=P State0=P
PORTB=0x0F;
DDRB=0x00;

// Port C initialization
// Func7=Out Func6=Out Func5=Out Func4=Out Func3=Out Func2=Out Func1=Out Func0=Out
// State7=0 State6=0 State5=0 State4=0 State3=0 State2=0 State1=0 State0=0
PORTC=0x00;
DDRC=0xFF;

// Port D initialization
// Func7=In Func6=In Func5=In Func4=In Func3=In Func2=Out Func1=Out Func0=Out
// State7=T State6=T State5=T State4=T State3=T State2=0 State1=0 State0=0
PORTD=0x00;
DDRD=0x07;

// Timer/Counter 0 initialization
// Clock source: System Clock
// Clock value: Timer 0 Stopped
// Mode: Normal top=FFh
// OC0 output: Disconnected
TCCR0=0x00;
TCNT0=0x00;
OCR0=0x00;

// Timer/Counter 1 initialization
// Clock source: System Clock
// Clock value: Timer 1 Stopped
// Mode: Normal top=FFFFh
// OC1A output: Discon.
// OC1B output: Discon.
// Noise Canceler: Off
// Input Capture on Falling Edge
// Timer 1 Overflow Interrupt: Off
// Input Capture Interrupt: Off
// Compare A Match Interrupt: Off
// Compare B Match Interrupt: Off
TCCR1A=0x00;
TCCR1B=0x00;
TCNT1H=0x00;
TCNT1L=0x00;
ICR1H=0x00;
ICR1L=0x00;
OCR1AH=0x00;
OCR1AL=0x00;
OCR1BH=0x00;
OCR1BL=0x00;

// Timer/Counter 2 initialization
// Clock source: System Clock
// Clock value: Timer 2 Stopped
// Mode: Normal top=FFh
// OC2 output: Disconnected
ASSR=0x00;
TCCR2=0x00;
TCNT2=0x00;
OCR2=0x00;

// External Interrupt(s) initialization
// INT0: Off
// INT1: Off
// INT2: Off
MCUCR=0x00;
MCUCSR=0x00;

// Timer(s)/Counter(s) Interrupt(s) initialization
TIMSK=0x00;

// Analog Comparator initialization
// Analog Comparator: Off
// Analog Comparator Input Capture by Timer/Counter 1: Off
ACSR=0x80;
SFIOR=0x00;

while (1)
{
PORTC = segment_decoder(inp());
PORTD = 1;
delay_us(10);
PORTD = 0;
PORTC = segment_decoder(inp());
PORTD = 2;
delay_us(10);
PORTD = 0;
PORTC = segment_decoder(inp());
PORTD = 4;
delay_us(10);
PORTD = 0;

// Place your code here

};
}


пробовал делать через switch - не получилось, через прерывание по таймеру - не получилось. все время горят нули, реакции на кнопки нет ни какой. где я ошибся?
Go to the top of the page
 
+Quote Post
aaarrr
сообщение Apr 25 2008, 18:38
Сообщение #2


Гуру
******

Группа: Свой
Сообщений: 10 713
Регистрация: 11-12-04
Пользователь №: 1 448



Цитата(Lost_Viking @ Apr 25 2008, 21:15) *
где я ошибся?

Вы ждете от порта определенного значения (PINB==0xFE), а не проверяете конкретный бит. Откуда уверенность, что в старших четырех битах именно 0xF?

Еще надо добавить антидребезг, замедлитель какой-нибудь и т.п.
Go to the top of the page
 
+Quote Post
Amper25
сообщение Apr 25 2008, 19:18
Сообщение #3


Местный
***

Группа: Свой
Сообщений: 215
Регистрация: 10-04-07
Пользователь №: 26 929



Один из самых простых вариантов:

Делатся прерывание или таймер на 10мсек. Берется 2 переменные.
KEYS_NOW и KEYS_PREV.
Каждые 10мсек, KEYS_NOW переписывается в KEYS_PREV,
а в KEYS_NOW записывается текущее положение кнопок.

Теперь если бите из KEYS_PREV записан 0 а в бите KEYS_NOW 1, то произошло нажатие кнопки. Таким образом получается и некое подобие антидребезга.


Цитата
if (PINB==0xFE) e++; //прибавляем единицы


Вот что первое бросается в глаза.

Надо заменить на:
if ((PINB & 0x01) == 0) e++;

Так как состояние на входах PINB 4..7 - абсолютно неясно.
Go to the top of the page
 
+Quote Post
Flasher
сообщение Apr 25 2008, 20:48
Сообщение #4


Местный
***

Группа: Свой
Сообщений: 374
Регистрация: 6-09-05
Из: Тирасполь, Приднестровье
Пользователь №: 8 294



У Вас при инициализации ясно видно что порт B, а именно старшие 4 разряда инициализируются как выходы с нулем. Естественно, что PINB.7 PINB.6 PINB.5 PINB.4 будут равны 0. А вы ждете и проверяете так, как будто они равны 1. Измените условие на проверку if(!PINB.3) if(!PINB.2) if(!PINB.1) if(!PINB.0)
Go to the top of the page
 
+Quote Post
Lost_Viking
сообщение Apr 25 2008, 21:00
Сообщение #5


Частый гость
**

Группа: Участник
Сообщений: 168
Регистрация: 25-04-08
Пользователь №: 37 091



ребят, спасибо. сейчас попробую. Кстати, пробовал ставить порт B полностью на вход - тоже самое получалось.


симулирую в PROTEUS.
Go to the top of the page
 
+Quote Post
Flasher
сообщение Apr 25 2008, 21:20
Сообщение #6


Местный
***

Группа: Свой
Сообщений: 374
Регистрация: 6-09-05
Из: Тирасполь, Приднестровье
Пользователь №: 8 294



Цитата(Lost_Viking @ Apr 26 2008, 00:00) *
ребят, спасибо. сейчас попробую. Кстати, пробовал ставить порт B полностью на вход - тоже самое получалось.
симулирую в PROTEUS.

Кстати, а ничего что функция Inp возвращает посчитанное число 0-999,которое она даже не помнит, а сегмент декодер воспринимает число от 0 до 9.?
Go to the top of the page
 
+Quote Post
Lost_Viking
сообщение Apr 26 2008, 13:39
Сообщение #7


Частый гость
**

Группа: Участник
Сообщений: 168
Регистрация: 25-04-08
Пользователь №: 37 091



Цитата(Flasher @ Apr 26 2008, 01:20) *
Кстати, а ничего что функция Inp возвращает посчитанное число 0-999,которое она даже не помнит, а сегмент декодер воспринимает число от 0 до 9.?


А я этого и не заметил smile.gif)))))))))
Go to the top of the page
 
+Quote Post
Lost_Viking
сообщение Apr 26 2008, 21:57
Сообщение #8


Частый гость
**

Группа: Участник
Сообщений: 168
Регистрация: 25-04-08
Пользователь №: 37 091



Цитата(Amper25 @ Apr 25 2008, 23:18) *
Один из самых простых вариантов:

Делатся прерывание или таймер на 10мсек. Берется 2 переменные.



попытался сделать прерывание по сравнению с таймером 0. хотел добиться динамической индикации с частотой 50гц,но PROTEUS мигал со слишком маленькой частотой. исходил я из следующих соображений:

частота чипа = 1 МГц.
нужно получить 50 гц.

выбираем делитель на 1024

1000000/1024 = 976.5625

Цитата
TCCR0=0x05;


976.5625/50 = 19.5 (Коэф. пересчета). округляем до 19

Цитата
OCR0=0x13;


врубаем прерывание по совпадению
Цитата
TIMSK=0x02;


вот, собственно что по прерыванию делаю:

Цитата
interrupt [TIM0_COMP] void timer0_comp_isr(void)
{
if (i==3) i=0; //всего 3 разряда.
PORTC=Dig[num()]; //выводим в PORTC значений из массива Dig, определяемое функцией num
switch (i) { //смотрим какое i
case 0: //нулевой разряд
PORTD.0=0; //зажигаем нулевой разряд, а все остальное гасим
PORTD.1=1; //и т.д.
PORTD.2=1;
break;
case 1:
PORTD.0=1;
PORTD.1=0;
PORTD.2=1;
break;
case 2:
PORTD.0=1;
PORTD.1=1;
PORTD.2=0;
break;}
i++; //переходим к следующему разряду при очередном вызове прерывания.

}


вроде бы все хорошо, даже кнопочки работают, и числа отображаются. но частота "развертки" очень маленькая.

пытался в OCR0 записать меньшее значение, что бы быстрее наступило совпадение (и тем самым быстрее вызвалось прерывание), но изменений не увидел.

может быть я что-то не понимаю? или просто Code Vision что-то не то делает , или PROTEUS?

где грабли?
Go to the top of the page
 
+Quote Post
Lost_Viking
сообщение Apr 26 2008, 23:25
Сообщение #9


Частый гость
**

Группа: Участник
Сообщений: 168
Регистрация: 25-04-08
Пользователь №: 37 091



все, врубился. оказывается, что после срабатывания прерывания по совпадению TCNT0 нужно было обнулить.
Go to the top of the page
 
+Quote Post
Lost_Viking
сообщение Apr 28 2008, 17:50
Сообщение #10


Частый гость
**

Группа: Участник
Сообщений: 168
Регистрация: 25-04-08
Пользователь №: 37 091



Цитата(Amper25 @ Apr 25 2008, 23:18) *
Один из самых простых вариантов:

Делатся прерывание или таймер на 10мсек. Берется 2 переменные.
KEYS_NOW и KEYS_PREV.
Каждые 10мсек, KEYS_NOW переписывается в KEYS_PREV,
а в KEYS_NOW записывается текущее положение кнопок.


Т.е. примрено так:

char num (void) {
if ((KEYS_PREV!=KEYS_NOW)&(PINB.0==0)) e++;
if (e==10) e=0;
if ((KEYS_PREV!=KEYS_NOW)&(PINB.1==0)) d++;
if (d==10) d=0;
if ((KEYS_PREV!=KEYS_NOW)&(PINB.2==0)) s++;
if (s==10) s=0;
if ((KEYS_PREV!=KEYS_NOW)&(PINB.3==0)) {
e=0;
d=0;
s=0;};
............

}

interrupt [TIM0_COMP] void timer0_comp_isr(void)
{
KEYS_PREV=KEYS_NOW; // как раз о чем и говорилось
KEYS_NOW=PINB; // ...
TCNT0=0xFF;
if (i==3) i=0;
PORTC=Dig[num()]; // вызываем функцию num
..... ;}


да,и подскажите: правильно ли я рассчитал значение, котокрое помещаем в OCR0:

Частота процессора = 1МГц.
Делитель = 1024
На таймер после делителя = 0,977 КГц, т.е. 977 Гц
Требуемая частота = 50Гц, т.е. 20 мс
Делим 977/50 = 19.54, округляем до 19. в hex = 0х13

TCCR0=0x05;
TCNT0=0x00;
OCR0=0x13;

при вызове обработчика прерывания обнуляем TCNT0.

Если все правильно, то почему в PROTEUS какое-то непонятное мигание?

вот файлики для протеуса и для CodeVision

Сообщение отредактировал Lost_Viking - Apr 28 2008, 17:46
Прикрепленные файлы
Прикрепленный файл  temp.rar ( 43.73 килобайт ) Кол-во скачиваний: 66
 
Go to the top of the page
 
+Quote Post
vet
сообщение Apr 29 2008, 10:11
Сообщение #11


Знающий
****

Группа: Свой
Сообщений: 550
Регистрация: 16-06-04
Из: Казань
Пользователь №: 32



19.54 я бы всё-таки округлил до 20. Но в OCR0 записывается верхнее значение счетчика, так что пишется в этом случае 20-1=19.
если воспользуетесь режимом CTC таймера, отпадет необходимость обнулять счетчик, и в точности отсчета интервалов выиграете, хоть оно в данном случае и не принципиально.


--------------------
Главная линия этого опуса ясна мне насквозь!
Go to the top of the page
 
+Quote Post
Lost_Viking
сообщение Apr 29 2008, 13:57
Сообщение #12


Частый гость
**

Группа: Участник
Сообщений: 168
Регистрация: 25-04-08
Пользователь №: 37 091



Цитата(vet @ Apr 29 2008, 14:11) *
19.54 я бы всё-таки округлил до 20. Но в OCR0 записывается верхнее значение счетчика, так что пишется в этом случае 20-1=19.
если воспользуетесь режимом CTC таймера, отпадет необходимость обнулять счетчик, и в точности отсчета интервалов выиграете, хоть оно в данном случае и не принципиально.


благодарю. теперь осталось все это собрать, и запустить. А то что-то PROTEUS динамическую индикацию криво симулирует.

Еще вопросец: какой номинал должен быть у емкости, подключенной к Aref, если я использую внутренний ИОН 2.56В?

Сообщение отредактировал Lost_Viking - Apr 29 2008, 14:06
Go to the top of the page
 
+Quote Post

Reply to this topicStart new topic
1 чел. читают эту тему (гостей: 1, скрытых пользователей: 0)
Пользователей: 0

 


RSS Текстовая версия Сейчас: 19th July 2025 - 04:04
Рейтинг@Mail.ru


Страница сгенерированна за 0.01452 секунд с 7
ELECTRONIX ©2004-2016