|
|
  |
Глобальный счетчик милисекунд. |
|
|
|
Jun 11 2018, 04:27
|
Профессионал
    
Группа: Участник
Сообщений: 1 778
Регистрация: 29-03-12
Пользователь №: 71 075

|
Есть програма для Kinetis K10. В ней есть глобальный счетчик милисекунд который инкрементируется каждую милисекунду в прерывании. Код void TI1_OnInterrupt(LDD_TUserData *UserDataPtr) { globalSysTimer++; } И на основе этой переменной строятся таймауты и периодические циклы. Это очень удобно и работать с переменной просто но мне не нравиться парадигма прерывания програмы каждую милисекунду. У Kinetis есть аппаратный таймер (FlexTimer Module (FTM)) 16 бит - это дает нам 65536 милисекунд до переполнения что конечно очень мало для нормальной работы. Что поинтересней можно придумать для Cortex-M4?
|
|
|
|
|
Jun 11 2018, 07:34
|
Гуру
     
Группа: Свой
Сообщений: 2 724
Регистрация: 14-05-07
Из: Ярославль, Россия
Пользователь №: 27 702

|
Цитата(Jenya7 @ Jun 11 2018, 07:27)  Это очень удобно и работать с переменной просто но мне не нравиться парадигма прерывания програмы каждую милисекунду. Ничего страшного. Я обычно делаю 10 мс, но при помощи SysTick. В Cortex именно этот таймер используется для этих целей. Цитата(Jenya7 @ Jun 11 2018, 07:27)  это дает нам 65536 милисекунд до переполнения что конечно очень мало для нормальной работы. А что значит "нормальная работа"? Переполнения конечно же нужно обрабатывать. При 32-битном счетчике переполнения будут каждые 49 дней, но их тоже нужно обрабатывать.
|
|
|
|
|
Jun 12 2018, 11:14
|
Гуру
     
Группа: Свой
Сообщений: 2 724
Регистрация: 14-05-07
Из: Ярославль, Россия
Пользователь №: 27 702

|
Цитата(Jenya7 @ Jun 12 2018, 12:24)  покажите где тут делитель Цитата The processor has a 24-bit system timer, SysTick, that counts down from the reload value to zero, reloads, that is wraps to, the value in the SYST_RVR register on the next clock edge, then counts down on subsequent clocks. Перевожу: есть 24-битный вычитающий таймер. Когда он досчитает до нуля, то перегружается значением из SYST_RVR. Таким образом можно получить периодические прерывания, а в них уже работать с софтовыми интервалами. Обращаю внимание на бит CLKSOURCE регистра SYST_CSR. Он выбирает источник тактирования. У STM32, например, можно тактировать от HCLK или HCLK / 8 за счет этого бита. Цитата The RCC feeds the Cortex System Timer (SysTick) external clock with the AHB clock (HCLK) divided by 8. The SysTick can work either with this clock or with the Cortex clock (HCLK), configurable in the SysTick Control and Status Register.
|
|
|
|
|
Jun 13 2018, 05:57
|
Гуру
     
Группа: Свой
Сообщений: 5 228
Регистрация: 3-07-08
Из: Омск
Пользователь №: 38 713

|
Цитата(adnega @ Jun 12 2018, 16:17)  или мили- секунд? Миля - единица длины, секунда - времени. Даже как-то затрудняюсь вот так ставить между ними знак '-'.  Цитата(adnega @ Jun 12 2018, 16:25)  У ТС именно "мили". Нужно понять что это значит. С точки зрения физики: значение разницы расстояния и времени пока не исследовано  Цитата(Jenya7 @ Jun 13 2018, 08:39)  все так грустно? только мне на планете Земля нужны временные метки в проекте? нет. Только у Вас это вызывает трудности
|
|
|
|
|
Jun 13 2018, 06:41
|
Профессионал
    
Группа: Участник
Сообщений: 1 778
Регистрация: 29-03-12
Пользователь №: 71 075

|
Цитата(AlexandrY @ Jun 13 2018, 12:02)  У Kinetis есть 64-х битный lifetimer на базе PIT. Так что хватит на всю жизнь с точностью в 16 ns PIT не имеет пределителя. его невозможно настроить на 1 милисекундный тик. можно через второй таймер но второй таймер генерирует прерывание - а я этого хочу избежать. Код // Timer 2 PIT_LDVAL2 = 0x00000009; // setup Timer 2 for 10 counts PIT_TCTRL2 = TIE; // enable Timer 2 interrupt PIT_TCTRL2 |= CHN; // chain Timer 2 to Timer 1 PIT_TCTRL2 |= TEN; // start Timer 2 // Timer 1 PIT_LDVAL1 = 0x23C345FF; // setup Timer 1 for 600 000 000 cycles PIT_TCTRL1 = TEN; // start Timer 1
Сообщение отредактировал Jenya7 - Jun 13 2018, 06:56
|
|
|
|
|
Jun 13 2018, 09:00
|
Профессионал
    
Группа: Участник
Сообщений: 1 778
Регистрация: 29-03-12
Пользователь №: 71 075

|
Цитата(ViKo @ Jun 13 2018, 14:25)  Допустим, тактовая частота процессора 120 МГц. На вход в прерывание и выход тратим по 12 тактов. На инкремент счетчика, пусть, еще 12 тактов. Итого: 36 / 120 000 = 0,0003 Вы боитесь потерять производительность на 0,03%?  я не боюсь потерять производительность. я не хочу чтоб у меня прерывание молотило кажую милисекунду. если вам это не мешает то вы счастливый человек.  Цитата(Сергей Борщ @ Jun 13 2018, 14:03)  Что мешает считать в тиках таймера? это типа шутка?
|
|
|
|
|
Jun 13 2018, 09:35
|
Гуру
     
Группа: Свой
Сообщений: 2 724
Регистрация: 14-05-07
Из: Ярославль, Россия
Пользователь №: 27 702

|
Jenya7, вы по все видимости начинающий разработчик - прислушайтесь к советам опытных МК-программистов. Зачем вам аппаратный таймер, если вы от него даже прерываний боитесь? Это две крайности. Нужно сделать аппаратный таймер для счета программного счетчика с интервалом 1 или 10 мс. Затем все миллисекундные задержки и выше делать программно. Только так. Если у вас какая-то неприязнь к такому решению, то вы в чем-то заблуждаетесь - можете высказать свои "фи", и мы вас переубедим. Повторюсь, у Cortex-ов есть специальный таймер SysTick, и 99% проектов использует его для миллисекундных и выше задержек, с программным отсчетом времени. Задержки на микросекунды и ниже уже могут потребовать аппаратный таймер. Использование аппаратного таймера жестко привязывает проект к данной архитектуре и данному камню - проект будет слабо переносимым и склонным к аппаратным конфликтам с другими библиотеками и т.п. Насколько я понимаю, ТС хочет сделать некую инициализацию, чтоб потом вычитывать из некого регистра число, равное числу миллисекунд. Желательно, чтоб регистр был 32-битным. Вроде, у LPC17xx таймеры были 32 битными с 32-битным предделителем. Цитата(Jenya7 @ Jun 13 2018, 12:25)  у меня есть критические секции. можно сказать очень критические. я не могу их сделать атомик, мне другие прерывания важны. У Cortex-M есть NVIC, который может делать вложенные прерывания с приоритетами. Если 32-битная переменная выровненная, то доступ к ней будет атомарным. Но я так понял, что из счетчика миллисекунд бы будете только вычитывать.
|
|
|
|
|
Jun 13 2018, 09:41
|
Профессионал
    
Группа: Участник
Сообщений: 1 778
Регистрация: 29-03-12
Пользователь №: 71 075

|
adnega так что вы предлагаете? я честно говоря так и не понял. по моему вот этого Цитата Нужно сделать аппаратный таймер для счета программного счетчика с интервалом 1 или 10 мс. я и пытаюсь добиться.
|
|
|
|
|
Jun 13 2018, 10:15
|
Гуру
     
Группа: Свой
Сообщений: 2 724
Регистрация: 14-05-07
Из: Ярославль, Россия
Пользователь №: 27 702

|
Цитата(Jenya7 @ Jun 13 2018, 12:41)  по моему вот этогоя и пытаюсь добиться. Дык, настраиваете SysTick на генерацию прерываний с интервалом 1 мс, а в обработчике прерывания делаете Код maintimer++; , где maintimer - 32-битная переменная, в которой будут лежать миллисекунды.
|
|
|
|
|
Jun 13 2018, 10:17
|
Профессионал
    
Группа: Участник
Сообщений: 1 778
Регистрация: 29-03-12
Пользователь №: 71 075

|
Цитата(adnega @ Jun 13 2018, 16:15)  Дык, настраиваете SysTick на генерацию прерываний с интервалом 1 мс, а в обработчике прерывания делаете Код maintimer++; , где maintimer - 32-битная переменная, в которой будут лежать миллисекунды. мда... ну вобщем тему можно закрывать. признаюсь был не прав.
|
|
|
|
|
Jun 17 2018, 03:37
|
Частый гость
 
Группа: Участник
Сообщений: 91
Регистрация: 3-07-11
Пользователь №: 66 028

|
Раньше, тоже делал счет миллисекунд из прерывания, но последнее время стал использовать такой способ http://we.easyelectronics.ru/Soft/samyy-pr...nyy-taymer.htmlТолько максимальное значение миллисекунд, которое можно задать зависит от таймера. А для сильно большой задержки, использую RTC, настроил что бы счетчик работал по секундам и считываю его, также. (только это относиться для stm32f1x, для stm32f4x RTC совсем другая )
|
|
|
|
|
Jun 18 2018, 05:02
|
Профессионал
    
Группа: Участник
Сообщений: 1 778
Регистрация: 29-03-12
Пользователь №: 71 075

|
Цитата(pokk @ Jun 17 2018, 08:37)  Раньше, тоже делал счет миллисекунд из прерывания, но последнее время стал использовать такой способ http://we.easyelectronics.ru/Soft/samyy-pr...nyy-taymer.htmlТолько максимальное значение миллисекунд, которое можно задать зависит от таймера. А для сильно большой задержки, использую RTC, настроил что бы счетчик работал по секундам и считываю его, также. (только это относиться для stm32f1x, для stm32f4x RTC совсем другая ) у меня таймера не хватало. в Кинетесе они кривые. кроме LPTM. освободил LPTM. он 16-битный приходиться приращивать в прерывании Код void LPTMR0_IRQHandler(void) { LPTMR0_CSR |= LPTMR_CSR_TCF_MASK; ovf_count++; added_val = 0xFFFF * ovf_count; }
uint32_t SYSTIME_GetSystemTime(void) { unsigned long cnt; LPTMR0_CNR = 0; // write any value to the counter register so that it puts its present counter value into a temporary register cnt = LPTMR0_CNR; // read the value from the temporary register return (cnt + added_val); }
Сообщение отредактировал Jenya7 - Jun 18 2018, 05:03
|
|
|
|
|
Jun 21 2018, 05:12
|
Профессионал
    
Группа: Участник
Сообщений: 1 778
Регистрация: 29-03-12
Пользователь №: 71 075

|
Цитата(adnega @ Jun 18 2018, 11:40)  А почему SysTick не используете? объяснил это в первом посте. Цитата(juvf @ Jun 20 2018, 09:42)  Этого мало? Это позволяет делать/контролировать задержки до 65 секунд без какого либо контроля переполнения, в лоб, без бубнов и танцев. Если нужно делать задержки измеряемые в секундах (например 1000 секунд, или 10000000 секунд), то можно отдельный таймер аппаратный завести с тактом 1 мс и переполнением после 1000 мс, в прерывании ++globalSec . Или вообще использовать RTC. у него нет пределителя - я не могу создать тик с периодом и 1 миллисекунду. Цитата(Obam @ Jun 20 2018, 23:54)  В K10 4-канальный 32-битный PIT (программируемый интервальный таймер) есть - делай счётчики интервалов какие хошь то же самое - у него нет пределителя - я не могу создать тик с периодом и 1 миллисекунду. Цитата(Obam @ Jun 20 2018, 23:54)  Впрочем, ТС сознался, что был не прав. долгие годы игры в покер научили меня вовремя скидывать слабую руку.
Сообщение отредактировал Jenya7 - Jun 21 2018, 05:14
|
|
|
|
|
Jun 21 2018, 05:15
|
Гуру
     
Группа: Свой
Сообщений: 2 724
Регистрация: 14-05-07
Из: Ярославль, Россия
Пользователь №: 27 702

|
Цитата(Jenya7 @ Jun 21 2018, 08:12)  у него нет пределителя Вы опять за старое. У него есть период, и за счет периода вы создадите 1 мс. С периодом 1 мс у вас будет шагать не таймер, а вызываться обработчик прерывания, в котором будет to_1ms++. Для этого подойдет любой таймер, но есть же специальный SysTick. Почему его не используют некоторые?
|
|
|
|
|
Jun 21 2018, 05:24
|

Профессионал
    
Группа: Свой
Сообщений: 1 261
Регистрация: 14-05-09
Из: Челябинск
Пользователь №: 49 045

|
Цитата(adnega @ Jun 21 2018, 10:15)  Вы опять за старое. У него есть период, и за счет периода вы создадите 1 мс. С периодом 1 мс у вас будет шагать не таймер, а вызываться обработчик прерывания, в котором будет to_1ms++. Писал же ТС, что у него сейчас так и есть, но он не хочет раз в 1 мс падать в прерывание, а хочет сделать аппаратный миллисекундный таймер. Цитата(Jenya7 @ Jun 21 2018, 10:12)  у него нет пределителя - я не могу создать тик с периодом и 1 миллисекунду. vs Цитата 16 бит - это дает нам 65536 милисекунд до переполнения как это? 16 битный таймер дает переполнение через 65536 милисекунд, но тик у этого таймера не равен 1 мс??? Проблема решена, и слава богу!!!
|
|
|
|
|
Jun 21 2018, 07:05
|
Гуру
     
Группа: Свой
Сообщений: 2 724
Регистрация: 14-05-07
Из: Ярославль, Россия
Пользователь №: 27 702

|
Цитата(juvf @ Jun 21 2018, 09:37)  А они мне зачем? У меня есть полностью аппаратный таймер, зачем мне программный таймер и обработчик прерывания? Допускаю, что есть эксклюзивные задачи, в которых одного таймера до 65 мс достаточно, но в общем случае вариант с 1мс-прерыванием гораздо универсальнее. В некоторых библиотеках я вообще пришел к двум/трем функциям on_init() - один раз при инициализации МК on_mainloop() - внутри mainloop`а on_timer() - внутри 10мс-таймера Решение гибкое, воспроизводимое на множестве камней и архитектур. Никаких дополнительный аппаратных требований, типа таймер с предделителем таким-то и т.п. Я не против исключений из правил, но их не стоит выдавать за правило. Я на atmega8 вообще без аппаратных таймеров делал программный UART-TX тупо nop-ами формировал задержку, и работало-ведь, но я же объявляю это нормой. Сделать софтовый 1мс таймер настолько просто, что грех от этого отказываться. И мне не понятно зачем? ТС говорил, что у него какие-то критические секции есть... думаю, больное место там.
|
|
|
|
|
Jun 21 2018, 07:26
|

Профессионал
    
Группа: Свой
Сообщений: 1 261
Регистрация: 14-05-09
Из: Челябинск
Пользователь №: 49 045

|
Цитата(adnega @ Jun 21 2018, 12:05)  Допускаю, что есть эксклюзивные задачи, в которых одного таймера до 65 мс достаточно, 65536 мс. не 65 мс, а 65 ТЫСЯЧ мс, это 65 секунд. И таймер один, но одновременно всяких интервалов и периодов можно задавать хоть сотню (как и на программном). Решение гибкое, воспроизводимое на множестве камней и архитектур. Никаких дополнительный аппаратных требований программных обработчиков, программного кода, настроек NVIC и т.п. Если зашли в критическую секцию (остановили прерывания), то таймер продолжит считать. Цитата тупо nop-ами формировал задержку .... я же объявляю это нормой опечатка? Цитата И мне не понятно зачем? Не знаю как объяснить? зачем в МК аппаратная перефирия? можно всё программно сделать, без таймеров, на ноп-ах и ногодрыгах.
|
|
|
|
|
Jun 21 2018, 09:12
|
Гуру
     
Группа: Свой
Сообщений: 2 724
Регистрация: 14-05-07
Из: Ярославль, Россия
Пользователь №: 27 702

|
Цитата(juvf @ Jun 21 2018, 10:26)  зачем в МК аппаратная перефирия? можно всё программно сделать, без таймеров, на ноп-ах и ногодрыгах. Я за полную загрузку периферии. В Cortex все унифицировали: есть одинаковый для всех SysTick и одинаковый NVIC. За это переживать не надо. Никогда не приходилось запрещать прерывания, а задачи решал самые разные. Для этого есть NVIC с ключевой буквой N, типа вложенных прерываний с гибкой настройкой приоритетов. Недавно обсуждали SVC вместо критических секций.
|
|
|
|
|
Jun 21 2018, 16:08
|

Знающий
   
Группа: Участник
Сообщений: 756
Регистрация: 14-11-14
Пользователь №: 83 663

|
Цитата долгие годы игры в покер научили меня вовремя скидывать слабую руку. Долгие годы программирования "железа" должны были научить пользоваться документацией на это "железо"  Цитата(Jenya7 @ Jun 21 2018, 08:12)  то же самое - у него нет пределителя - я не могу создать тик с периодом и 1 миллисекунду. RefMan на подсемейство K10: про PIT гл. 41 (всего 12 стр) разд. 41.4.3 волшебные слова "Chain mode" - тик будет с любым периодом. Редко у чего бывает "предделитель" в 32 разряда.
--------------------
Пролетарий умственного труда.
|
|
|
|
|
  |
1 чел. читают эту тему (гостей: 1, скрытых пользователей: 0)
Пользователей: 0
|
|
|