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

 
 
 
Reply to this topicStart new topic
> mega8 timer..., Прерывания 16-bit таймера по сравнению.
BVU
сообщение Aug 17 2005, 10:01
Сообщение #1


Профессионал
*****

Группа: Свой
Сообщений: 1 301
Регистрация: 30-11-04
Из: Россия, Н.Новгород
Пользователь №: 1 264



Всем, привет!
Для mega8 пытаюсь сделать прерывание по сравнению - используя 16-bit timer-B. Выполняю инициализацию, таймер начинает работать увеличивая программный счетчик в теле своего прерывания, но работа идет явно не по сравнению константного значения, а как бы по переполнению... Почему такие выводы спросите вы, потому что при изменении константы в регисте сравнения в два раза - частота прерываний не меняется... sad.gif. По всей видимости где-то, что-то я не доглядел или непонял.
Привожу пример своего кода:

void InitTimers(void)
{
/* Initialization Timer1 */
OCR1B = QUARTER_sec; // Set compare value on one second.
TIMSK = (1 << OCIE1B); // Set mask to enable Timer1B compare interrupt.
TCCR1B = ((1 << CS12) | (1 << CS10)); // Set prescaller on 1024.
}


// IRQ TIMER1_COMPB
#pragma vector = TIMER1_COMPB_vect
__interrupt void timer1_compB_Processing(void)
{
char pd;

__disable_interrupt();
tick++;
... // обработка подпрограммы.
__enable_interrupt();
}

Ваши мнения корифеи программирования таймеров???

С уважением,
BVU


--------------------
Не корысти ради, не в целях наживы, а во исполнение велений души!
Go to the top of the page
 
+Quote Post
she
сообщение Aug 17 2005, 10:21
Сообщение #2





Группа: Новичок
Сообщений: 5
Регистрация: 10-08-05
Пользователь №: 7 507



Цитата(BVU @ Aug 17 2005, 14:01)
/* Initialization Timer1 */
    OCR1B = QUARTER_sec;    // Set compare value on one second.
    TIMSK = (1 << OCIE1B);    // Set mask to enable Timer1B compare interrupt.
    TCCR1B = ((1 << CS12) | (1 << CS10));    // Set prescaller on 1024.


*


а режим в битах WGM10:3 дядя устанавливать будет?=)
Go to the top of the page
 
+Quote Post
prottoss
сообщение Aug 17 2005, 10:52
Сообщение #3


Гуру
******

Группа: Свой
Сообщений: 2 720
Регистрация: 24-03-05
Пользователь №: 3 659



Цитата(she @ Aug 17 2005, 18:21)
Цитата(BVU @ Aug 17 2005, 14:01)

/* Initialization Timer1 */
    OCR1B = QUARTER_sec;    // Set compare value on one second.
    TIMSK = (1 << OCIE1B);    // Set mask to enable Timer1B compare interrupt.
    TCCR1B = ((1 << CS12) | (1 << CS10));    // Set prescaller on 1024.


*


а режим в битах WGM10:3 дядя устанавливать будет?=)
*



Если таймер работает просто в режиме счетчика, а не в режиме WGM, то эти биты трогать не надо


--------------------
Go to the top of the page
 
+Quote Post
prottoss
сообщение Aug 17 2005, 10:59
Сообщение #4


Гуру
******

Группа: Свой
Сообщений: 2 720
Регистрация: 24-03-05
Пользователь №: 3 659



Цитата(BVU @ Aug 17 2005, 18:01)
Всем, привет!
Для mega8 пытаюсь сделать прерывание по сравнению  - используя 16-bit timer-B. Выполняю инициализацию, таймер начинает работать увеличивая программный счетчик в теле своего прерывания, но работа идет явно не по сравнению константного значения, а как бы по переполнению... Почему такие выводы спросите вы, потому что при изменении константы в регисте сравнения в два раза - частота прерываний не меняется... sad.gif.  По всей видимости где-то, что-то я не доглядел или непонял.
Привожу пример своего кода:

void InitTimers(void)
{
/* Initialization Timer1 */
    OCR1B = QUARTER_sec;    // Set compare value on one second.
    TIMSK = (1 << OCIE1B);    // Set mask to enable Timer1B compare interrupt.
    TCCR1B = ((1 << CS12) | (1 << CS10));    // Set prescaller on 1024.
}
*


Если Вы меняете константу сравнения, то частота прерываний не изменится, если Вы не установите бит WGM12(СTC) (сброс по совпадению ТС1) - но он работает только для OCR1A, а Ваш код для OCR1В. По этому частота прерываний менятся не будет. Вы только смещаете во времени начало прерывания от сброса счетчика в 0x0000.
Мой Вам совет - проэмулируйте все в AVRStudio - и все всанет на свои места.

Цитата(BVU @ Aug 17 2005, 18:01)
// IRQ TIMER1_COMPB
#pragma vector = TIMER1_COMPB_vect
__interrupt void timer1_compB_Processing(void)
{
char pd;

    __disable_interrupt();
    tick++;
    ...  // обработка подпрограммы.
    __enable_interrupt();
}

Ваши мнения корифеи программирования таймеров???

С уважением,
BVU
*


__disable_interrupt(); __enable_interrupt(); - это не нужно, при входе в прерывние бит глобальных прерываний автоматически сбрасывается. Так что это пустое.


--------------------
Go to the top of the page
 
+Quote Post
BVU
сообщение Aug 17 2005, 13:20
Сообщение #5


Профессионал
*****

Группа: Свой
Сообщений: 1 301
Регистрация: 30-11-04
Из: Россия, Н.Новгород
Пользователь №: 1 264



По всей видимости после прерывания на совпадение счетчик TCNT1 продолжает счет до 0xffff и сбросившись в 0x0000 работа продолжается до следующего совпадения. Тем самым создается иллюзия, что таймер работает по переполнению и не зависит от константы сравнения.
В этом случае в самом начале обработки прерывания таймера придется обнулять TCNT1 "в ручную" (высокой точности периода мне пока не требуется, ну а если возникнет такая необходимость ее можно компенсировать ассемблерными вставками - nop) или воспользоваться советом prottoss.

Тем не менее всем участникам спасибо за помощь!

С уважением,
BVU


--------------------
Не корысти ради, не в целях наживы, а во исполнение велений души!
Go to the top of the page
 
+Quote Post
yung
сообщение Aug 17 2005, 18:33
Сообщение #6


Местный
***

Группа: Свой
Сообщений: 207
Регистрация: 25-03-05
Из: Рязань
Пользователь №: 3 669



Лучше будет использовать режим СТС - сброс при совпадении. Почитай pdf, там это есть.
Go to the top of the page
 
+Quote Post
prottoss
сообщение Aug 17 2005, 19:39
Сообщение #7


Гуру
******

Группа: Свой
Сообщений: 2 720
Регистрация: 24-03-05
Пользователь №: 3 659



Цитата(yung @ Aug 18 2005, 02:33)
Лучше будет использовать режим СТС - сброс при совпадении. Почитай pdf, там это есть.
*


Только, в таком случае, не забудь в качестве регистра по совпадению использовать OCR1А


--------------------
Go to the top of the page
 
+Quote Post
arttab
сообщение Aug 18 2005, 01:28
Сообщение #8


Профессионал
*****

Группа: Свой
Сообщений: 1 432
Регистрация: 7-12-04
Из: Новосибирск
Пользователь №: 1 371



Сделайте как сказано выше - на OCR1А. Проверено - работает.
OCR1A = 0x001E;
TCCR1B = 0x0D; //0D no filt. capt. wave. 8E6/1024 (00000101)=781.5
и соответственно #pragma vector=TIMER1_COMPA_vect
Сам разбирался в даташите и чтобы развеять сомнения воспользовался мастеров в кодевижене.


--------------------
OrCAD, Altium,IAR, AVR....
Go to the top of the page
 
+Quote Post

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

 


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


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