Помощь - Поиск - Пользователи - Календарь
Полная версия этой страницы: Помогите разобраться с таймером
Форум разработчиков электроники ELECTRONIX.ru > Микроконтроллеры (MCs) > MSP430
LCD
Использую А0 для формирования задержек, а А1 для часов.
Вот так я инициализирую таймер:
Код
  TACCTL0=CCIE;
  TACCTL1=CCIE;
  TACCR1=32767;
  TACTL=TASSEL0+TAIE+MC0;
  //IAR показывает, что TACCTL0=0x418, TACCTL1=0x10

А0 работает так, как от него требуется.
Код
void sleep(unsigned short tacts)
{
  __enable_interrupt();
  TACCR0=tacts;
  __low_power_mode_3();
  TACCR0=0;
}

#pragma vector=TIMERA0_VECTOR
__interrupt void TimerA0_ISR(void)
{
  __low_power_mode_off_on_exit ();
};

А что происходит с А1, я понять не могу: он перестает генерировать прерывания в LPM3. Вроде, все делаю, как написано в исходниках-примерах.
Код
#pragma vector=TIMERA1_VECTOR
__interrupt void TimerA1_ISR(void)
{
  if(TAIV==2)
  {
    (здесь увеличиваем время на секунду)
  }
  P2OUT ^= 0x40; //мигаем светодиодом
}

Помогите разобраться, где здесь ошибка.
rezident
У вас как минимум две ошибки.
Во-первых, вы используете режим таймера CountUp (MC0=1, MC1=0) при котором таймер считает вверх (инкрементируется содержимое счетчика TAR) лишь до значения, установленного в TACCR0. После совпадения TAR и TACCR0 счетчик TAR сбрасывается в нуль. Поэтому в режиме CountUp запись в TACCR1 какого либо значения меньшего, чем записано в TACCR0 не дает никакого эффекта. Потому, что при таких условиях (TACCR1>TACCR0) прерывание от совпадения TAR и TACCR1 никогда не наступит.
Во-вторых, в этом режиме запись в TACCT0 нуля дает останов таймера.
В-третьих, у вас присутствует прерывание от переполнения (установлен бит TAIE в TACTL), которое вы не обрабатываете.
Для функций которые вы желаете реализовать таймер должен работать в режиме Continuous - режим счета с переполнением. Поскольку таймер тактируется от часового кварца 32768Гц, то прерывание дважды за период переполнения может вызываться 1) при совпадении какого-либо из CCR и 2) прерыванием от переполнения. Так получем два прерывания за 65536 тактов или период 1 секунду.
Обработчик прерывания выглядит так
Код
#pragma vector=TIMERA1_VECTOR
__interrupt void TimerA1_ISR(void)
{ unsigned int TAIVstate=TAIV;
  switch(TAIVstate)
  { case 0x02:  //прерывания при совпадении TAR и TACCR1
    case 0x0A:  //прерывания при переполнении TAR
      //вызывается дважды за период переполнения (2 с), т.е. с периодом 1 секунда
      //(здесь увеличиваем время на секунду)
      break;
    default:
      break;
  }
  P2OUT ^= 0x40; //мигаем светодиодом
}

Инициализацию таймера делаем так
Код
TACTL=TASSEL0 | TAIE | TACLR;  //здесь разрешаем первый источник прерываний за период
TACCR1=32767;
TACCTL0=0;
TACCTL1=CCIE; //здесь разрешаем второй источник прерываний
TACCTL2=0;
TACTL|=MC1;

Для реализации функции интервальных отсчетов используем TACCR0 как это у вас и задумано. Только не нужно его сбрасывать. Когда вы запускаете интервальный отсчет, то просто возмите текущее значение TAR, добавьте к нему требуемое значение интервала в периодах частоты часового кварца, запишите получившееся число в TACCR0 и разрешите прерывание от совпадения TAR и TACCR0 установкой бита CCIE в TACCTL0, предварительно очистив флаг CCIFG.
Код
void sleep(unsigned int tacts)
{
  TACCTL0=CM0 | CMIS1 | CAP;
  TACCTL0|=CMIS0;
  TACCR0+=tacts;
  TACCTL0=CCIE;
  __bis_SR_register(__SR_CPU_OFF | __SR_SCG0 | __SR_SCG1 | GIE);
  TACCTL0=0;
}

#pragma vector=TIMERA0_VECTOR
__interrupt void TimerA0_ISR(void)
{
  __bic_SR_register_on_exit(__SR_CPU_OFF | __SR_SCG0 | __SR_SCG1);
};

Обращаю особо внимание на три первые строчки функции sleep. Предполагаю, что для MCLK у вас используется DCO, так? Тогда поскольку частоты MCLK=DCO и ACLK=LFXT (используемая для тактирования таймера A) асинхронные, то впрямую считывать TAR нельзя. Прямое чтение TAR может дать неверное значение. Чтобы этого избежать используем режим захвата, сымитировав перепад уровня для его срабатывания. Вот это и реализовано в первых двух строчках. В третьей строке "захваченное" в регистр TACCR0 текущее значение TAR увеличивается на интервал времени, соответствующий требуемому. Естественно интервал не может превышать периода 2 сек или 65536 тактов таймера или увеличению на 0x10000=0x0000, т.к. регистр 16-и разрядный.
LCD
Спасибо за подробнейшее разъяснение. Вставил ваш код - все заработало.
keks9357
вот такой вопрос.
нужно что бы каждые 2 секунды было прерывание по таймеру и сброс счетчика n.
делаю так:

#pragma vector=TIMERA0_VECTOR
__interrupt void TimerA0_ISR(void)
{
unsigned TAIVcopy=TAIV;
switch(TAIVcopy)
{
case 0x02:break;
case 0x0A:break;
}
n=0; //n - счетчик нажатий
}

void main()

TACCTL1=CCIE;
TACCR1 = 32767;
TACTL = TASSEL_1 +MC_1 +TACLR+ TAIE;

_EINT();

while(1)
{

while (!(P2IN&0x20));

while (P2IN&0x20);

n++;

out(v); //вывод количества нажатий на экран

}
}

А в итоге на экране всего лишь общее количество нажатий.в чем запарка?
MrYuran
Цитата(keks9357 @ Oct 7 2011, 15:04) *
А в итоге на экране всего лишь общее количество нажатий.в чем запарка?

volatile SomeType n;
скорее всего.
По листингу можно сказать точно.
Psych
А после void main() открывающей скобки { точно нету? biggrin.gif
rezident
Цитата(keks9357 @ Oct 7 2011, 16:04) *
в чем запарка?
Во-первых, как уже отметил MrYuran, переменная n должна иметь квалификатор volatile. Во-вторых, немного странно ожидать прерывание в функции, расположенной по адресу вектора TIMERA0_VECTOR, в то время, как разрешены прерывания от событий совпадения TAR и CCR1 и от переполнения TAR. Ведь оба этих источника прерываний используют другой вектор - TIMERA1_VECTOR.
keks9357
1. volatile int n;
2. поменял вектор прерывания TIMERA0_VECTOR на TIMERA1_VECTOR;
3. прерываний по совпадению и переполнению счетчика как небыло так и нет;
4. и на экране выводит количество нажатий
5. после void main {...} скобочка есть
rezident
А часовой кварц у вас вообще генерит? То бишь ACLK присутствует? И таймер запускается?
По крайней мере стоит исправить команды инициализации таймера также, как они приведены у меня в сообщении №2
Код
TACTL = TASSEL_1 | TACLR | TAIE;
TACCR1 = 32767;
TACCTL1 = CCIE;
TACTL |= MC_1;

Обратите внимание, что установка бита TACLR влияет (сбрасывает их) на биты MC (count direction) и ID (clock divider). Поэтому в одной команде одновременно с TACLR их устанавливать не имеет смысла - после выполнения команды биты MC и ID все равно будут сброшены.
Цитата
TACLR Bit 2 Timer_A clear. Setting this bit resets TAR, the clock divider, and the count direction. The TACLR bit is
automatically reset and is always read as zero
keks9357
спасибо всем заработало. 1111493779.gif
подскажите идейку как сделать спидометр, как на велосипедах.
rezident
Цитата(keks9357 @ Oct 12 2011, 09:57) *
подскажите идейку как сделать спидометр, как на велосипедах.

Использовать таймер в режиме захвата (capture).
Вообще посмотрите ez430-chronos. Беспроводную связь можете не использовать. Используйте корпус и ЖКИ от него. Там практически все готово для спидометра.
FREEKER
Цитата(rezident @ Oct 12 2011, 17:06) *
Использовать таймер в режиме захвата (capture).
Вообще посмотрите ez430-chronos. Беспроводную связь можете не использовать. Используйте корпус и ЖКИ от него. Там практически все готово для спидометра.


Я не использовал capture. Определял таймер в режиме UP. По прерыванию порта P1 смотрю TAR, вычисляю, обнуляю, а переполнение счётчика по прерыванию таймера смотрю, как доходит до конца, прибавляю к переменной единичку и т.д.
rezident
Цитата(FREEKER @ Oct 18 2011, 12:18) *
Я не использовал capture. Определял таймер в режиме UP. По прерыванию порта P1 смотрю TAR, вычисляю, обнуляю, а переполнение счётчика по прерыванию таймера смотрю, как доходит до конца, прибавляю к переменной единичку и т.д.
В режиме capture таймер выполняет все те же самые функции, что вы описали, но только аппаратно. wink.gif
keks9357
какие вычисления делаете? и зачем прибавлять к счетчику переполнений единицу?
rezident
Цитата(keks9357 @ Oct 25 2011, 15:03) *
зачем прибавлять к счетчику переполнений единицу?

Типично - для (программного) расширения разрядности счетчика. Когда аппаратных 16-и бит не хватает для вычисления периода.
keks9357
Реализовать при 12МГц задержку в 3 секунды возможно? если можно пример laugh.gif
rezident
Цитата(keks9357 @ Oct 27 2011, 10:57) *
Реализовать при 12МГц задержку в 3 секунды возможно? если можно пример laugh.gif
Да хоть на 300 секунд. wink.gif Примерно так.
CODE
#include <msp430x24x.h>
#include <stdint.h>

#define FREQ_TACLK 12000000UL //TimerA clock frequency
#define LED_OUT (1U<<0) //P1.0 - LED output
#define PULSE_IN (1U<<2) //P1.2 - Pulse Input (CCI1A)
#define DELAY_LEDOUT (FREQ_TACLK*3UL) //Value ticks of TimerA = 3s*12MHz

int main(void)
{ WDTCTL = WDTPW | WDTHOLD;
/* Basic Clock module */
BCSCTL3 = LFXT1S_3;
DCOCTL = CALDCO_12MHZ;
BCSCTL1 = CALBC1_12MHZ | XT2OF;
BCSCTL2 = 0;
/* Port1 input/output */
P1DIR = LED_OUT;
P1SEL = PULSE_IN;
P1OUT = 0;
/* TimerA */
TACTL = TASSEL_2 | TACLR | TAIE;
TACCTL0 = 0;
TACCTL1 = CM_3 | CCIS_0 | SCS | CAP | CCIE;
TACCTL2 = 0;
TACTL |= MC_2;

__enable_interrupt();

for (;;)
{
__bic_SR_register(LPM0_bits);
__no_operation();
}
}

#pragma vector=TIMERA1_VECTOR
#pragma type_attribute=__interrupt
void TIMERA_ISR(void)
{ static uint32_t timeCntr, timeDly;
static uint16_t flag;
uint16_t CCRVal;

switch(TAIV)
{ case 0x02: //TACCR1 vector, capture P1.2
CCRVal=TACCR1;
timeDly=timeCntr + DELAY_LEDOUT + CCRVal;
flag = 1;
TACCTL1 &= ~CCIE;
break;
case 0x04: //TACCR2 vector, compare
P1OUT ^= LED_OUT;
TACCTL2 &= ~CCIE;
TACCTL1 |= CCIE;
break;
case 0x0A: //TAR overflow vector, overflow counter
timeCntr += 1UL<<16;
if (flag != 0)
{ if ((timeDly - timeCntr) < (1UL<<16))
{ CCRVal=(uint16_t)(timeDly - timeCntr);
TACCR2 = CCRVal;
TACCTL2 |= CCIE;
flag = 0;
}
}
break;
default:
break;
}
}

P1.0 - выход, к которому подключен светодиод.
P1.2 - вход управления.
По любому перепаду уровня (1->0 или 0->1) на входе P1.2 через установленное время (в примере - 3 сек) светодиод на выходе P1.0 переключится в противоположное состояние. В это время вход P1.2 нечувствителен к перепадам уровня.
Весь функционал реализован на аппаратуре таймера A.
Используется вход захвата CCI1A (P1.2 для MSP430F24x), перепад на котором вызывает захват (capture) значения TAR и запись его в CCR1. При этом в преывании вычисляется величина временной отметки timeDly, отстоящая от данного события на установленное время задержки (3 сек в отсчетах таймера на частоте 12МГц).
CCR2 используется в режиме сравнения (compare) для более точного задания времени задержки.
Прерывание от переполнения используется для расширения разрядности таймера. По переполнению 32-х разрядный счетчик timeCntr инкрементируется на величину разрядности таймера (65536). При установленном флаге flag текущее значение timeDly сравнивается с временной отметкой timeCntr. Если разница укладывается в разрядность таймера (меньше, чем 65536), то остаток разности заносится в CCR2 и разрешается прерывание при событии совпадения TAR и CCR2.
Далее при наступлении события совпадения TAR и CCR2 в прерывании переключается состояние светодиода на пине P1.0 и вновь активируется вход управления.
Точность установки задержки примерно в 20 тактов, которые можно учесть при ее вычислении.
В этом примере есть один нюанс, влияющий на точность задержки, про который я пока умолчу. Если (тщательно изучая мануал) поймете и правильно сформулируете вопрос, то поясню как эту багу можно обойти. sm.gif
P.S. в железе не проверялось laughing.gif
keks9357
excl.gif rezident - гуру excl.gif
Nathan Stark
Скажите, какой минимальный набор команд нужен, чтобы Таймер А считал в инкрементальном режиме до определенного числа?

Исходя из документации делаю так:
TACCR0 = 0x100; //Задаем модуль счета таймера
TACTL = 0x110; //Задаем MC0 = 1, то есть запускаем счетчик в режиме "вверх",
//Задаем TASSEL0 = 1, то есть задаем источник импульсов.

Но в итоге регистр TAR никак не меняется. Кажется я чего-то не понимаю
rezident
Цитата(Nathan Stark @ Dec 19 2011, 20:22) *
Исходя из документации делаю так:
TACCR0 = 0x100; //Задаем модуль счета таймера
TACTL = 0x110; //Задаем MC0 = 1, то есть запускаем счетчик в режиме "вверх",
//Задаем TASSEL0 = 1, то есть задаем источник импульсов.

Неправильно. Сначала следует проинициализировать регистры таймера (TASSEL и TACCR0) и только потом запускать счет, установив бит MC0 в TACTL. Т.е. минимально три команды. Хотя я предпочитаю все значащие регистры проинициализировать так, чтобы исключить "случайно возникающие" прерывания (от CCR0, CCR1, CCR2), которые программой не предусмотрены.
Код
TACTL = TASSEL0 | TACLR; // ACLK/1 в качестве входного клока, сброс TAR, прерывание от переполнения запрещено
TACCR0 = 0x100 - 1; //без единицы, т.к. состояние 0x0000 тоже считается
TACCTL0 = 0; //режим сравнения, запрет прерывания от CCR0
TACCTL1 = 0; //режим сравнения, запрет прерывания от CCR1
TACCTL2 = 0; //режим сравнения, запрет прерывания от CCR2
TACTL |= MC0; //запуск счета, режим CountUp

Цитата(Nathan Stark @ Dec 19 2011, 20:22) *
Но в итоге регистр TAR никак не меняется. Кажется я чего-то не понимаю
В железе или в симуляторе? В симуляторе IAR периферия не симулируется!
Nathan Stark
Цитата
В железе или в симуляторе? В симуляторе IAR периферия не симулируется!


Оу! А как же тогда проверить работу алгоритма на логику если нет железа?
ILYAUL
Цитата(Nathan Stark @ Dec 19 2011, 19:58) *
Оу! А как же тогда проверить работу алгоритма на логику если нет железа?


Спаять макетку и проверять
Nathan Stark
Ага, понятно) Спасибо за помощь)
Еще такой вопрос, а в прерывания по таймеру программа тоже заходить не будет?
rezident
Цитата(Nathan Stark @ Dec 19 2011, 21:15) *
Еще такой вопрос, а в прерывания по таймеру программа тоже заходить не будет?
В симуляторе можно "вручную" имитировать вызов прерываний Simulator -> Forced Interrupt или настроить макрос для симуляции вызова Simulator -> Interrupt Setup.
Для просмотра полной версии этой страницы, пожалуйста, пройдите по ссылке.
Invision Power Board © 2001-2025 Invision Power Services, Inc.